设计模式学习-装饰模式

定义

装饰模式又叫包装模式,可以动态的给一个对象添加一些额外的职责。装饰模式比生成子类更灵活,是继承关系的替代方案之一。

UML类图

装饰设计模式

如上,主要有被装饰者和装饰者角色,分别如下

  • Component

    抽象组件,被装饰对象的抽象,可以是接口或者抽象类。

  • ConcreteComponent

    Component的实现类,具体被装饰的对象。

  • Decoator

    抽象装饰者,该类继承自Component组件,并且内部持有一个组件对象的引用。负责装饰组件对象。

  • ConcreteDecorator

    具体的装饰者,负责实现具体的装饰方法。

示例

装饰模式最简单常见的例子就是人的着装打扮。我们可以每天穿不同的衣服对自己进行打扮,但是被打扮的人还是不变的,我们还是我们,只是穿了不同的衣服。

/**
 * 抽象组件Component,被装饰的原始类
 * 定义抽象的人
 */
public abstract class People {
    public abstract void dress();
}

/**
 * 实现类ConcreteComponent,具体要被装饰的类
 * 定义具体的男孩
 */
public class Boy extends People {
    @Override
    public void dress() {
        System.out.println("男孩穿衣服");
    }
}


/**
 * 抽象装饰类Decoator
 * 该类中保存了一个被装饰类的引用,方便调用被装饰类的方法
 * 定义衣服的抽象
 */
public abstract class PersonDecorate extends People {
    private People people;

    public PersonDecorate(People people) {
        this.people = people;
    }

    @Override
    public void dress() {
        people.dress();
    }
}

/**
 * 具体装饰类ConcreteDecorator
 * 添加实现具体的扩展功能,比如这里是dressWorkClothe方法
 * 并且在原方法中整合调用
 */
public class WorkClothes extends PersonDecorate {
    public WorkClothes(People people) {
        super(people);
    }

    @Override
    public void dress() {
        super.dress();
        dressWorkClothe();
    }

    private void dressWorkClothe(){
        System.out.println("穿职业装");
    }
}
/**
 * 客户端使用
 * 实例化Boy对象,并将其传入到WorkClothes对象中
 * 调用WorkClothes的dress方法
 */
  public static void main(String args[]){
        Boy boy = new Boy();
        WorkClothes workClothes = new WorkClothes(boy);
        workClothes.dress();
    }
    男孩穿衣服
    穿职业装

Android源码中的装饰模式

我们在安卓开发中最常使用的Context类就是装饰模式的经典实现。

首先看一下Context类

public abstract class Context {
    //...
    /**
     * Same as {@link #startActivity(Intent, Bundle)} with no options
     * specified.
     *
     * @param intent The description of the activity to start.
     *
     * @throws ActivityNotFoundException  
     *`
     * @see #startActivity(Intent, Bundle)
     * @see PackageManager#resolveActivity
     */
    public abstract void startActivity(@RequiresPermission Intent intent);
    
    //...
}

Context中定义了大量我们平时开发中常用的方法,例如startActivity等,但是Context是一个抽象类。其具体实现是由ContextImpl完成的

class ContextImpl extends Context {
    @Override
    public void startActivity(Intent intent) {
        warnIfCallingFromSystemProcess();
        startActivity(intent, null);
    }
    @Override
    public void startActivity(Intent intent, Bundle options) {
        warnIfCallingFromSystemProcess();

        // Calling start activity from outside an activity without FLAG_ACTIVITY_NEW_TASK is
        // generally not allowed, except if the caller specifies the task id the activity should
        // be launched in. A bug was existed between N and O-MR1 which allowed this to work. We
        // maintain this for backwards compatibility.
        final int targetSdkVersion = getApplicationInfo().targetSdkVersion;

        if ((intent.getFlags() & Intent.FLAG_ACTIVITY_NEW_TASK) == 0
                && (targetSdkVersion < Build.VERSION_CODES.N
                        || targetSdkVersion >= Build.VERSION_CODES.P)
                && (options == null
                        || ActivityOptions.fromBundle(options).getLaunchTaskId() == -1)) {
            throw new AndroidRuntimeException(
                    "Calling startActivity() from outside of an Activity "
                            + " context requires the FLAG_ACTIVITY_NEW_TASK flag."
                            + " Is this really what you want?");
        }
        mMainThread.getInstrumentation().execStartActivity(
                getOuterContext(), mMainThread.getApplicationThread(), null,
                (Activity) null, intent, -1, options);
    }
}

ContextImpl就相当于装饰模式中的抽象组件的具体实现类。我们知道Activity本质上也是Context,看一下Activity的继承结构

public class Activity extends ContextThemeWrapper
        implements LayoutInflater.Factory2,
        Window.Callback, KeyEvent.Callback,
        OnCreateContextMenuListener, ComponentCallbacks2,
        Window.OnWindowDismissedCallback, WindowControllerCallback,
        AutofillManager.AutofillClient {
            // ...
        }

Activity并没有直接继承自Context,而是继承自ContextThemeWrapper,而ContextThemeWrapper又继承自ContextWrapper,ContextWrapper最终继承自Context

public class ContextWrapper extends Context {
    Context mBase;

    public ContextWrapper(Context base) {
        mBase = base;
    }
    
    @Override
    public void startActivity(Intent intent) {
        mBase.startActivity(intent);
    }
  // ...
}      

这里ContextWrapper对象持有Context实现类ContextImpl对象的一个引用mBase,调用了 mBase.startActivity(intent)方法。那么这个ContextWrapper对象就是装饰者对象,并且调用了被装饰者ContextImpl的方法。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,287评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,346评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,277评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,132评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,147评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,106评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,019评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,862评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,301评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,521评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,682评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,405评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,996评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,651评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,803评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,674评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,563评论 2 352

推荐阅读更多精彩内容

  • Android插件化基础的主要内容包括 Android插件化基础1-----加载SD上APKAndroid插件化基...
    隔壁老李头阅读 4,590评论 2 35
  • 设计模式概述 在学习面向对象七大设计原则时需要注意以下几点:a) 高内聚、低耦合和单一职能的“冲突”实际上,这两者...
    彦帧阅读 3,741评论 0 14
  • 装饰模式也叫做包装模式,是结构型设计模式之一。目的是为了给一个类或对象增加行为。可以是继承的一种替代。 装饰模式也...
    喵了个呜s阅读 1,843评论 3 6
  • Android Context 基类源码/frameworks/base/core/java/android/co...
    southtrain阅读 4,123评论 0 5
  • 距离婚礼还有13天,总觉得要写点什么来承上启下。 打开电脑后,记忆却像墨水一样晕开变淡。没办法投影13年前第一次见...
    Moi_凹特曼阅读 305评论 0 0