JavaEE进阶知识学习-----设计模式---装饰者模式

定义

动态的将责任附加到对象上,若要扩展功能,装饰者模式提供了比继承更具有弹性的方案。

为什么会出现这个设计模式

给对象扩展行为的方法有两种,一种是通过继承,继承是给类添加扩展行为比较有效的办法,通过使用继承,可以使得子类有自己的行为,还可以获得父类的行为方法,但是使用继承是静态的,在编译的时候就已经决定了子类有哪些行为。
当然还可以使用关联,将一个对象嵌入到另一个对象中,有一个对象来决定是否引用该对象来扩展自己的行为,这是一种动态的方式,我们可以在程序中动态的决定和控制。
前面所说的两种都会导致一种‘类爆炸’的情况出现,所以就出现了装饰者模式。

认识装饰者模式

例如为咖啡店设计一个点咖啡的程序,采用饮料为主体,在运行时以调料来‘装饰’饮料,如果顾客要摩卡和奶泡深焙咖啡,那么,

  1. 拿一个深焙咖啡(DarkRoast)对象
  2. 以摩卡(Mocha)对象装饰它
  3. 一奶泡(Whip)对象装饰它
  4. 调用cost方法,并依赖委托(delegate)将调料的钱加上去。

装饰者类图结构

image

实现装饰者模式

情景:购买咖啡时,会加入不同的调料,根据不同的调料来收费,也就是说不同的咖啡与不同的调料有N中不同的组合方式,也就是出现了不同组合就应该有不同的价格。结构图如下:

image

代码实现

Beverage组件基类

public abstract class Beverage {
    protected String description = "";
    public String getDescription() {
        return description;
    }
    public abstract double cost();

}

HouseBlend组件

public class HouseBlend extends Beverage {
    public HouseBlend() {
        description = "这是一杯综合咖啡";
    }
    @Override
    public double cost() {
        return 0.89;
    }

}

Espresso组件

public class Espresso extends Beverage{
    public Espresso() {
        description = "这是一杯浓缩咖啡";
    }
    @Override
    public double cost() {
        return 1.99;
    }

}

Decat组件

public class Decat extends Beverage {
    public Decat() {
        description = "这是一杯深焙咖啡";
    }
    @Override
    public double cost() {
        return 0.99;
    }

}

DarkRoast

public class DarkRoast extends Beverage{
    public DarkRoast() {
        description = "这是一杯低咖啡因咖啡";
    }
    @Override
    public double cost() {
        return 1.05;
    }

}

配料基本类

public abstract class CondimentDecorator extends Beverage{
    public abstract String getDescription();
}

Milk配料

public class Milk extends CondimentDecorator{
    Beverage beverage;
    public Milk(Beverage beverage) {
        this.beverage = beverage;
    }
    @Override
    public String getDescription() {
        return beverage.getDescription()+",牛奶";
    }

    @Override
    public double cost() {
        return beverage.cost()+0.1;
    }

}

Mocha配料

public class Mocha extends CondimentDecorator {
    Beverage beverage;
    public Mocha(Beverage beverage) {
        this.beverage = beverage;
    }
    @Override
    public String getDescription() {
        return beverage.getDescription()+",摩卡";
    }

    @Override
    public double cost() {
        return beverage.cost()+0.2;
    }

}

Soy配料

public class Soy extends CondimentDecorator{
    Beverage beverage;
    public Soy(Beverage beverage) {
        this.beverage = beverage;
    }

    @Override
    public String getDescription() {
        return beverage.getDescription()+",豆浆";
    }

    @Override
    public double cost() {
        return beverage.cost()+0.15;
    }

}

Whip配料

public class Whip extends CondimentDecorator{
    Beverage beverage;
    public Whip(Beverage beverage) {
        this.beverage = beverage;
    }
    @Override
    public String getDescription() {
        return beverage.getDescription()+",奶泡";
    }

    @Override
    public double cost() {
        return beverage.cost()+0.1;
    }

}

测试类

public class StartbuzzCoffee {
    public static void main(String[] args) {
        Beverage beverage = new Espresso();
        System.out.println(beverage.getDescription()+",$"+beverage.cost());
        System.out.println("========================");
        
        Beverage beverage2 = new DarkRoast();
        beverage2 = new Mocha(beverage2);
        beverage2 = new Mocha(beverage2);
        beverage2 = new Whip(beverage2);
        System.out.println(beverage2.getDescription()+",$"+beverage2.cost());
    }

}

测试结果

这是一杯浓缩咖啡,$1.99
========================
这是一杯低咖啡因咖啡,摩卡,摩卡,奶泡,$1.55

总结

  1. 装饰者可以提供比继承更多的灵活性。
  2. 可以通过一种动态的方式来扩展一个对象的功能,在运行时选择不同的的装饰器,从而实现不同的行为。
  3. 具体组件类和装饰类可以独立变化,用户可以根据自己的需要增加具体的组件类和装饰类,原有的代码无需改变,负荷”开闭原则“。
  4. 但是也会产生很多的小对象,增加了系统的负责性。
  5. 建议在不影响其他对象的时候使用,以动态,透明的方式给单个对象添加职责。
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,445评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,889评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,047评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,760评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,745评论 5 367
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,638评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,011评论 3 398
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,669评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,923评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,655评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,740评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,406评论 4 320
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,995评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,961评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,197评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,023评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,483评论 2 342

推荐阅读更多精彩内容