• 【Android UI设计与开发】第14期:顶部标题栏(五)两种方式实现仿微信标题栏弹窗效果


    转载请注明出处:http://blog.csdn.net/yangyu20121224/article/details/9093821        

          博主在这篇文章中将会继续围绕顶部标题栏专题来进行实例讲解,今天要讲解的主题是分别使用PopupWindow和Activity两种不同的方式来实现仿微信顶部标题栏弹窗的这样一个效果。

    一、实现效果图

    这里为了演示方便,我将两种方法放在一个应用程序中演示,这个是主界面

    虽然两种实现的方式不一样,但是最终的效果图都是差不多的


     


     

    二、项目结构图

     


     

    三、详细的编码实现

     

    3.1 主界面的实现

    为了演示方便,我这里把两种实现方式分成两个Activity界面放在了主Activity界面中。

    1、主布局资源文件,activity_main.xml:

    <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
        xmlns:tools="http://schemas.android.com/tools"
        android:layout_width="match_parent"
        android:layout_height="match_parent" >
    
        <Button
            android:id="@+id/main_btn01"
            android:layout_width="fill_parent"
            android:layout_height="44dp"
            android:layout_above="@+id/main_btn02"
            android:layout_margin="5dp"
            android:background="@drawable/main_btn"
            android:text="第一种实现方式(PopupWindow实现)"
            android:textSize="16dp" />
    
        <Button
            android:id="@+id/main_btn02"
            android:layout_width="fill_parent"
            android:layout_height="44dp"
            android:layout_centerVertical="true"
            android:layout_margin="5dp"
            android:background="@drawable/main_btn"
            android:text="第二种实现方式(Activity实现)"
            android:textSize="16dp" />
    
    </RelativeLayout>

    2、定义一个自定义按钮的资源文件,main_btn.xml:

    <?xml version="1.0" encoding="utf-8"?>
    <selector xmlns:android="http://schemas.android.com/apk/res/android">
    
        <item android:drawable="@drawable/btn_back_pre" android:state_pressed="true"/>
        <item android:drawable="@drawable/btn_back_nor"/>
    
    </selector>

    3、主Activity程序入口类,MainActivity.java:

    package com.yangyu.mytitlebar01;
    
    import android.app.Activity;
    import android.content.Intent;
    import android.os.Bundle;
    import android.view.View;
    import android.view.View.OnClickListener;
    import android.widget.Button;
    
    /**
     * @author yangyu
     *    功能描述:主Activity类,程序的入口类
     */
    public class MainActivity extends Activity implements OnClickListener {
        //定义按钮
        private Button mainBtn01,mainBtn02;
        
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_main);
                            
            initView();        
        }
    
        /**
         * 初始化组件
         */
        private void initView(){
            //得到按钮并设置监听事件
            mainBtn01 = (Button)findViewById(R.id.main_btn01);
            mainBtn02 = (Button)findViewById(R.id.main_btn02);        
            
            mainBtn01.setOnClickListener(this);
            mainBtn02.setOnClickListener(this);
        }    
            
        @Override
        public void onClick(View v) {
            switch (v.getId()) {
            case R.id.main_btn01:
                startActivity(new Intent(MainActivity.this,CustomTitleActivity01.class));
                break;
            case R.id.main_btn02:
                startActivity(new Intent(MainActivity.this,CustomTitleActivity02.class));
                break;                
            default:
                break;
            }        
        }
        
    }

    3.2 第一种实现方式(PopupWindow)

     

    第一种实现方式主要是通过点击按钮来弹出一个PopupWindow菜单来实现的,步骤如下:

    1、标题栏的布局资源文件,这个资源文件在第二种实现方式中也会使用到,activity_main.xml:

    <?xml version="1.0" encoding="utf-8"?>
    <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
        android:layout_width="fill_parent"
        android:layout_height="fill_parent"
        android:background="#fcfcfc"
        android:orientation="vertical" >
    
        <RelativeLayout
            android:id="@+id/title"
            android:layout_width="fill_parent"
            android:layout_height="45dp"
            android:background="@drawable/title_bar"
            android:gravity="center_vertical" >
    
            <TextView
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:layout_centerInParent="true"
                android:text="微信"
                android:textColor="#ffffff"
                android:textSize="20sp" />
    
            <ImageButton
                android:id="@+id/title_btn"
                android:layout_width="67dp"
                android:layout_height="wrap_content"
                android:layout_alignParentRight="true"
                android:layout_centerVertical="true"
                android:layout_marginRight="5dp"
                android:background="@drawable/title_button"
                android:onClick="btnmainright"
                android:src="@drawable/title_btn_function" />
        </RelativeLayout>
    
    </LinearLayout>

    2、弹窗的布局页面,这里定义了一个ListView,title_popup.xml:

    <?xml version="1.0" encoding="utf-8"?>
    <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:background="@drawable/title_function_bg"
        android:orientation="vertical" >
    
        <ListView
            android:id="@+id/title_list"
            android:layout_width="120dp"
            android:layout_height="fill_parent"
            android:cacheColorHint="#00000000"
            android:divider="@drawable/mm_title_functionframe_line"
            android:listSelector="@drawable/title_list_selector"
            android:padding="3dp"
            android:scrollingCache="false" />
    
    </LinearLayout>

    3、定义一个列表选项中的自定义按钮,title_list_selector.xml:

    <?xml version="1.0" encoding="utf-8"?>
    <selector xmlns:android="http://schemas.android.com/apk/res/android">
    
        <item android:drawable="@drawable/mm_title_functionframe_pressed" android:state_focused="true"></item>
        <item android:drawable="@drawable/mm_title_functionframe_pressed" android:state_pressed="true"/>
        <item android:drawable="@drawable/mm_title_functionframe_pressed" android:state_selected="true"></item>
        <item android:drawable="@android:color/transparent"></item>
    
    </selector>

    4、最后再定义一个弹窗按钮的自定义按钮,title_button.xml:

    <?xml version="1.0" encoding="UTF-8"?>
    <selector xmlns:android="http://schemas.android.com/apk/res/android">
    
        <item android:drawable="@drawable/mm_title_btn_focused" android:state_focused="true"/>
        <item android:drawable="@drawable/mm_title_btn_pressed" android:state_pressed="true"/>
        <item android:drawable="@drawable/mm_title_btn_pressed" android:state_selected="true"/>
        <item android:drawable="@drawable/mm_title_btn_normal"/>
    
    </selector>

    5、下面是Java代码部分,首先定义一个常量类,Util.java:

    package com.yangyu.mytitlebar01;
    
    import android.content.Context;
    
    /**
     * @author yangyu
     *    功能描述:常量工具类
     */
    public class Util {
        /**
         * 得到设备屏幕的宽度
         */
        public static int getScreenWidth(Context context) {
            return context.getResources().getDisplayMetrics().widthPixels;
        }
    
        /**
         * 得到设备屏幕的高度
         */
        public static int getScreenHeight(Context context) {
            return context.getResources().getDisplayMetrics().heightPixels;
        }
    
        /**
         * 得到设备的密度
         */
        public static float getScreenDensity(Context context) {
            return context.getResources().getDisplayMetrics().density;
        }
    
        /**
         * 把密度转换为像素
         */
        public static int dip2px(Context context, float px) {
            final float scale = getScreenDensity(context);
            return (int) (px * scale + 0.5);
        }
    }

    6、再定义一个实体对象类,ActionItem这个类主要是用来绘制列表选项中的标题和图标,ActionItem.java:

    package com.yangyu.mytitlebar01;
    
    import android.content.Context;
    import android.graphics.drawable.Drawable;
    
    /**
     * @author yangyu
     *    功能描述:弹窗内部子类项(绘制标题和图标)
     */
    public class ActionItem {
        //定义图片对象
        public Drawable mDrawable;
        //定义文本对象
        public CharSequence mTitle;
        
        public ActionItem(Drawable drawable, CharSequence title){
            this.mDrawable = drawable;
            this.mTitle = title;
        }
        
        public ActionItem(Context context, int titleId, int drawableId){
            this.mTitle = context.getResources().getText(titleId);
            this.mDrawable = context.getResources().getDrawable(drawableId);
        }
        
        public ActionItem(Context context, CharSequence title, int drawableId) {
            this.mTitle = title;
            this.mDrawable = context.getResources().getDrawable(drawableId);
        }
    }

    7、再定义一个TitlePopup标题栏弹窗类,该类继承自PopupWindow,TitlePopup.java:

    package com.yangyu.mytitlebar01;
    
    import java.util.ArrayList;
    
    import android.content.Context;
    import android.graphics.Rect;
    import android.graphics.drawable.BitmapDrawable;
    import android.view.Gravity;
    import android.view.LayoutInflater;
    import android.view.View;
    import android.view.ViewGroup;
    import android.view.ViewGroup.LayoutParams;
    import android.widget.AdapterView;
    import android.widget.AdapterView.OnItemClickListener;
    import android.widget.BaseAdapter;
    import android.widget.ListView;
    import android.widget.PopupWindow;
    import android.widget.TextView;
    
    /**
     * @author yangyu
     *    功能描述:标题按钮上的弹窗(继承自PopupWindow)
     */
    public class TitlePopup extends PopupWindow {
        private Context mContext;
    
        //列表弹窗的间隔
        protected final int LIST_PADDING = 10;
        
        //实例化一个矩形
        private Rect mRect = new Rect();
        
        //坐标的位置(x、y)
        private final int[] mLocation = new int[2];
        
        //屏幕的宽度和高度
        private int mScreenWidth,mScreenHeight;
    
        //判断是否需要添加或更新列表子类项
        private boolean mIsDirty;
        
        //位置不在中心
        private int popupGravity = Gravity.NO_GRAVITY;    
        
        //弹窗子类项选中时的监听
        private OnItemOnClickListener mItemOnClickListener;
        
        //定义列表对象
        private ListView mListView;
        
        //定义弹窗子类项列表
        private ArrayList<ActionItem> mActionItems = new ArrayList<ActionItem>();            
        
        public TitlePopup(Context context){
            //设置布局的参数
            this(context, LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT);
        }
        
        public TitlePopup(Context context, int width, int height){
            this.mContext = context;
            
            //设置可以获得焦点
            setFocusable(true);
            //设置弹窗内可点击
            setTouchable(true);    
            //设置弹窗外可点击
            setOutsideTouchable(true);
            
            //获得屏幕的宽度和高度
            mScreenWidth = Util.getScreenWidth(mContext);
            mScreenHeight = Util.getScreenHeight(mContext);
            
            //设置弹窗的宽度和高度
            setWidth(width);
            setHeight(height);
            
            setBackgroundDrawable(new BitmapDrawable());
            
            //设置弹窗的布局界面
            setContentView(LayoutInflater.from(mContext).inflate(R.layout.title_popup, null));
            
            initUI();
        }
            
        /**
         * 初始化弹窗列表
         */
        private void initUI(){
            mListView = (ListView) getContentView().findViewById(R.id.title_list);
            
            mListView.setOnItemClickListener(new OnItemClickListener() {
                @Override
                public void onItemClick(AdapterView<?> arg0, View arg1, int index,long arg3) {
                    //点击子类项后,弹窗消失
                    dismiss();
                    
                    if(mItemOnClickListener != null)
                        mItemOnClickListener.onItemClick(mActionItems.get(index), index);
                }
            }); 
        }
        
        /**
         * 显示弹窗列表界面
         */
        public void show(View view){
            //获得点击屏幕的位置坐标
            view.getLocationOnScreen(mLocation);
            
            //设置矩形的大小
            mRect.set(mLocation[0], mLocation[1], mLocation[0] + view.getWidth(),mLocation[1] + view.getHeight());
            
            //判断是否需要添加或更新列表子类项
            if(mIsDirty){
                populateActions();
            }
            
            //显示弹窗的位置
            showAtLocation(view, popupGravity, mScreenWidth - LIST_PADDING - (getWidth()/2), mRect.bottom);
        }
        
        /**
         * 设置弹窗列表子项
         */
        private void populateActions(){
            mIsDirty = false;
            
            //设置列表的适配器
            mListView.setAdapter(new BaseAdapter() {            
                @Override
                public View getView(int position, View convertView, ViewGroup parent) {
                    TextView textView = null;
                    
                    if(convertView == null){
                        textView = new TextView(mContext);
                        textView.setTextColor(mContext.getResources().getColor(android.R.color.white));
                        textView.setTextSize(14);
                        //设置文本居中
                        textView.setGravity(Gravity.CENTER);
                        //设置文本域的范围
                        textView.setPadding(0, 10, 0, 10);
                        //设置文本在一行内显示(不换行)
                        textView.setSingleLine(true);
                    }else{
                        textView = (TextView) convertView;
                    }
                    
                    ActionItem item = mActionItems.get(position);
                    
                    //设置文本文字
                    textView.setText(item.mTitle);
                    //设置文字与图标的间隔
                    textView.setCompoundDrawablePadding(10);
                    //设置在文字的左边放一个图标
                    textView.setCompoundDrawablesWithIntrinsicBounds(item.mDrawable, null , null, null);
                    
                    return textView;
                }
                
                @Override
                public long getItemId(int position) {
                    return position;
                }
                
                @Override
                public Object getItem(int position) {
                    return mActionItems.get(position);
                }
                
                @Override
                public int getCount() {
                    return mActionItems.size();
                }
            }) ;
        }
        
        /**
         * 添加子类项
         */
        public void addAction(ActionItem action){
            if(action != null){
                mActionItems.add(action);
                mIsDirty = true;
            }
        }
        
        /**
         * 清除子类项
         */
        public void cleanAction(){
            if(mActionItems.isEmpty()){
                mActionItems.clear();
                mIsDirty = true;
            }
        }
        
        /**
         * 根据位置得到子类项
         */
        public ActionItem getAction(int position){
            if(position < 0 || position > mActionItems.size())
                return null;
            return mActionItems.get(position);
        }            
        
        /**
         * 设置监听事件
         */
        public void setItemOnClickListener(OnItemOnClickListener onItemOnClickListener){
            this.mItemOnClickListener = onItemOnClickListener;
        }
        
        /**
         * @author yangyu
         *    功能描述:弹窗子类项按钮监听事件
         */
        public static interface OnItemOnClickListener{
            public void onItemClick(ActionItem item , int position);
        }
    }

    8、最后再定义一个Activity界面类,CustomTitleActivity01.java:

    package com.yangyu.mytitlebar01;
    
    import android.app.Activity;
    import android.os.Bundle;
    import android.view.View;
    import android.view.View.OnClickListener;
    import android.view.ViewGroup.LayoutParams;
    import android.widget.ImageButton;
    
    /**
     * @author yangyu
     *    功能描述:第一种实现方式,PopupWindow实现方式
     */
    public class CustomTitleActivity01 extends Activity {
        //定义标题栏上的按钮
        private ImageButton titleBtn;
        
        //定义标题栏弹窗按钮
        private TitlePopup titlePopup;
    
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_title);
            
            initView();
            
            initData();
        }
        
        /**
         * 初始化组件
         */
        private void initView(){
            //实例化标题栏按钮并设置监听
            titleBtn = (ImageButton) findViewById(R.id.title_btn);
            titleBtn.setOnClickListener(new OnClickListener() {
                @Override
                public void onClick(View v) {
                    titlePopup.show(v);
                }
            });
                    
            //实例化标题栏弹窗
            titlePopup = new TitlePopup(this, LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT);
        }
        
        /**
         * 初始化数据
         */
        private void initData(){
            //给标题栏弹窗添加子类
            titlePopup.addAction(new ActionItem(this, "发起聊天", R.drawable.mm_title_btn_compose_normal));
            titlePopup.addAction(new ActionItem(this, "听筒模式", R.drawable.mm_title_btn_receiver_normal));
            titlePopup.addAction(new ActionItem(this, "登录网页", R.drawable.mm_title_btn_keyboard_normal));
            titlePopup.addAction(new ActionItem(this, "扫一扫",  R.drawable.mm_title_btn_qrcode_normal));
        }
        
    }

    3.2 第二种实现方式(Activity)

     

    第二种实现方式主要是通过再定义一个Activity来调用实现的,要想实现一个Activity点击按钮实现另一个Activity的弹窗

    效果,就要在AndroidManifest.xml清单文件中注册一个新的Activity,然后给它调用系统的样式来实现这种效果。

    <activity
                android:name="DialogActivity"
                android:theme="@android:style/Theme.Translucent.NoTitleBar" >
            </activity>

    1、定义另一个Activity布局文件,activity_dialog.xml:

    <?xml version="1.0" encoding="UTF-8"?>
    <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
        android:layout_width="fill_parent"
        android:layout_height="fill_parent" >
    
        <RelativeLayout
            android:layout_width="fill_parent"
            android:layout_height="fill_parent"
            android:layout_marginTop="50dp" >
    
            <LinearLayout
                android:id="@+id/main_dialog_layout"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:layout_alignParentRight="true"
                android:layout_alignParentTop="true"
                android:background="@drawable/title_function_bg"
                android:orientation="vertical" >
    
                <LinearLayout
                    android:id="@+id/llayout01"
                    android:layout_width="fill_parent"
                    android:layout_height="wrap_content"
                    android:layout_marginLeft="5dp"
                    android:layout_marginRight="5dp"
                    android:layout_marginTop="5dp"
                    android:background="@drawable/title_list_selector" >
    
                    <ImageView
                        android:id="@+id/imageView1"
                        android:layout_width="wrap_content"
                        android:layout_height="wrap_content"
                        android:layout_gravity="center_vertical"
                        android:layout_marginLeft="8dp"
                        android:src="@drawable/mm_title_btn_compose_normal" />
    
                    <TextView
                        android:layout_width="wrap_content"
                        android:layout_height="wrap_content"
                        android:padding="8dp"
                        android:text="发起聊天"
                        android:textColor="#fff"
                        android:textSize="16sp" />
                </LinearLayout>
    
                <ImageView
                    android:id="@+id/imageView5"
                    android:layout_width="wrap_content"
                    android:layout_height="wrap_content"
                    android:src="@drawable/mm_title_functionframe_line" />
    
                <LinearLayout
                    android:id="@+id/llayout02"
                    android:layout_width="fill_parent"
                    android:layout_height="wrap_content"
                    android:layout_marginLeft="5dp"
                    android:layout_marginRight="5dp"
                    android:background="@drawable/title_list_selector" >
    
                    <ImageView
                        android:id="@+id/imageView2"
                        android:layout_width="wrap_content"
                        android:layout_height="wrap_content"
                        android:layout_gravity="center_vertical"
                        android:layout_marginLeft="8dp"
                        android:src="@drawable/mm_title_btn_receiver_normal" />
    
                    <TextView
                        android:layout_width="wrap_content"
                        android:layout_height="wrap_content"
                        android:padding="8dp"
                        android:text="听筒模式"
                        android:textColor="#fff"
                        android:textSize="16sp" />
                </LinearLayout>
    
                <ImageView
                    android:id="@+id/imageView5"
                    android:layout_width="wrap_content"
                    android:layout_height="wrap_content"
                    android:src="@drawable/mm_title_functionframe_line" />
    
                <LinearLayout
                    android:id="@+id/llayout03"
                    android:layout_width="fill_parent"
                    android:layout_height="wrap_content"
                    android:layout_marginLeft="5dp"
                    android:layout_marginRight="5dp"
                    android:background="@drawable/title_list_selector" >
    
                    <ImageView
                        android:id="@+id/imageView3"
                        android:layout_width="wrap_content"
                        android:layout_height="wrap_content"
                        android:layout_gravity="center_vertical"
                        android:layout_marginLeft="8dp"
                        android:src="@drawable/mm_title_btn_keyboard_normal" />
    
                    <TextView
                        android:layout_width="wrap_content"
                        android:layout_height="wrap_content"
                        android:padding="8dp"
                        android:text="登录网页"
                        android:textColor="#fff"
                        android:textSize="16sp" />
                </LinearLayout>
    
                <ImageView
                    android:id="@+id/imageView5"
                    android:layout_width="wrap_content"
                    android:layout_height="wrap_content"
                    android:src="@drawable/mm_title_functionframe_line" />
    
                <LinearLayout
                    android:id="@+id/llayout04"
                    android:layout_width="fill_parent"
                    android:layout_height="wrap_content"
                    android:layout_marginBottom="3dp"
                    android:layout_marginLeft="5dp"
                    android:layout_marginRight="5dp"
                    android:background="@drawable/title_list_selector" >
    
                    <ImageView
                        android:id="@+id/imageView4"
                        android:layout_width="wrap_content"
                        android:layout_height="wrap_content"
                        android:layout_gravity="center_vertical"
                        android:layout_marginLeft="8dp"
                        android:src="@drawable/mm_title_btn_qrcode_normal" />
    
                    <TextView
                        android:layout_width="wrap_content"
                        android:layout_height="wrap_content"
                        android:padding="13dp"
                        android:text="扫一扫"
                        android:textColor="#fff"
                        android:textSize="16sp" />
                </LinearLayout>
            </LinearLayout>
        </RelativeLayout>
    
    </RelativeLayout>

    2、另一个Activity弹窗界面,DialogActivity.java:

    package com.yangyu.mytitlebar01;
    
    
    import android.app.Activity;
    import android.os.Bundle;
    import android.view.MotionEvent;
    import android.view.View;
    import android.view.View.OnClickListener;
    import android.widget.LinearLayout;
    
    /**
     * @author yangyu
     *    功能描述:弹出Activity界面
     */
    public class DialogActivity extends Activity implements OnClickListener{
        private LinearLayout layout01,layout02,layout03,layout04;
        
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_dialog);
    
            initView();
        }
    
        /**
         * 初始化组件
         */
        private void initView(){
            //得到布局组件对象并设置监听事件
            layout01 = (LinearLayout)findViewById(R.id.llayout01);
            layout02 = (LinearLayout)findViewById(R.id.llayout02);
            layout03 = (LinearLayout)findViewById(R.id.llayout03);
            layout04 = (LinearLayout)findViewById(R.id.llayout04);
    
            layout01.setOnClickListener(this);
            layout02.setOnClickListener(this);
            layout03.setOnClickListener(this);
            layout04.setOnClickListener(this);
        }
        
        @Override
        public boolean onTouchEvent(MotionEvent event){
            finish();
            return true;
        }
        
        @Override
        public void onClick(View v) {
            
        }
    }

    3、最后是第二种方式的显示界面,CustomTitleActivity02.java:

    package com.yangyu.mytitlebar01;
    
    import android.app.Activity;
    import android.content.Intent;
    import android.os.Bundle;
    import android.view.View;
    import android.view.View.OnClickListener;
    import android.widget.ImageButton;
    
    /**
     * @author yangyu
     *    功能描述:第二种实现方式,Activity实现方式
     */
    public class CustomTitleActivity02 extends Activity {
        //定义标题栏上的按钮
        private ImageButton titleBtn;
            
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_title);
            
            initView();            
        }
        
        /**
         * 初始化组件
         */
        private void initView(){
            //实例化标题栏按钮并设置监听
            titleBtn = (ImageButton) findViewById(R.id.title_btn);
            titleBtn.setOnClickListener(new OnClickListener() {
                @Override
                public void onClick(View v) {
                    startActivity(new Intent(CustomTitleActivity02.this,DialogActivity.class));
                }
            });                        
        }    
            
    }

    两种弹窗的实现方式基本上就讲完了,有什么问题可以跟博主留言。

    源码下载地址

  • 相关阅读:
    Docker-CentOS系统安装Docker
    Docker-准备Docker环境
    Docker系列-文章汇总
    商品订单库存一致性问题的思考
    java模板、工厂设计模式在项目中的重构
    2018Java年底总结
    java的AQS中enp没有同步代码块为啥是原子操作
    java使用awt包在生产环境docker部署时出现中文乱码的处理
    初探装饰器模式
    开灯问题
  • 原文地址:https://www.cnblogs.com/a354823200/p/4105452.html
Copyright © 2020-2023  润新知