设计模式——装饰模式

设计模式———装饰模式

例子:成绩单报告

在面向对象的设计中,如果超过两层继承,可能就出设计问题了。这是经验总结,并非定论。继承层次越多,维护成本越多。

成绩装饰模型
增加一个抽象类和两个实现类,其中Decorator的作用是封装SchoolReort类
抽象成绩单
//抽象成绩单
public abstract class SchoolReport{
  public abstract void report();
  public abstract void sign(String name);
}
具体的四年级成绩
public abstract FourthGradeSchoolReport extends SchoolReport{
  public void report(){
    System.out.println("尊敬的XX家长");
    System.out.println("成绩如下");
    System.out.println("语文xx");
    // ...
  }
  public void sign(String name){
    System.out.println("家长签名为:"+name);
  }
}
修饰的抽象类
// 修饰的抽象类
public abstract class Decorator extends SchoolReport{
  private SchoolReport sr;
  public Decorator(SchoolReport sr){
    this.sr = sr;
  }
  // 报告成绩
  public void report(){
    this.sr.report();
  }
  //家长签名
  public void sign(String name){
    this.sr.sign(name);
  }
}
最高成绩修饰
public class HighScoreDecorator extends Decorator  {
  public HighScoreDecorator(SchoolReport sr){
    super(sr);
  }
  private void reportHighScore(){
    System.out.println("最高成绩是:XX");
  }
  public void report(){
    this.reportHighScore();
    super.report();
  }
}
排名情况修饰
public class SortDecorator extends Decorator   {
  public SortDecorator(SchoolReport sr){
    super(sr);
  }
  public void reportSort(){
    System.out.println("排名情况是xx");
  }
  @Override
  public void report(){
    this.reportSort();
    super.report();
  }
}
查看成绩单
public class Father{
  public static void main(String[] args) {
    //把成绩单拿过来
    SchoolReport sr;
    //原装的成绩单
    sr = new FourthGradeSchoolReport();
    //加了最高分说明的成绩单
    sr = new HighScoreDecorator(sr);
    // 加了排名的说明
    sr = new SortDecorator(sr);
    //看成绩单
    sr.report();
    sr.sign("name");
  }
}

装饰模式的定义

Attach additional responsibilities to an object dynamically keeping the same interface. Decorators provide a flexible alternative to subclassing for extending functionality.

动态地给一个对象添加一些额外的职责。就增加功能来说,装饰模式相比生成子类更为灵活。

装饰模式
四个角色说明
  • component抽象构件
    • component是一个接口或者是抽象类,就是定义我们最核心的对象,也就是最原始的对象
    • 在装饰模式中,必然有一个最基本,最核心,也就是最原始的对象
  • concreteComponent具体构建
    • concreteComponent是最核心,最原始,最基本的接口或抽象类的实现,要装饰的就是它
  • Decorator装饰角色
    • 一般是一个抽象类,实现接口或者抽象方法
    • 它的属性里必定有一个private变量指向Component抽象构建
  • ConcreteDecorator具体装饰角色
    • ConcreteDecoratorA和ConcreteDecoratorB是两个具体的装饰类,要把最原始,最基本,最核心的进行装饰
抽象构件
public abstract class Component{
  //抽象的方法
  public abstract void operate();
}
具体构件
public class ConcreteComponent extends Component  {
  // 具体实现
  public void operate(){
    // code to do something
    System.out.println("doSomethin");
  }
}
抽象装饰者
public abstract class Decorator extends Component{
  //必然有一个变量是抽象构件
  private Component component = null;
  //通过构造函数传递被修饰者
  public Decorator(Component component){
    this.component = component;
  }
  //委托给被修饰者执行
  @Override
  public void operate(){
    this.component.operate();
  }
}
具体的装饰类
public class ConcreteDecorator1 extends Decorator{
  public ConcreteDecorator1(Component component){
    super(component);
  }
  //定义自己的修饰方法
  public void method(){
    //code to do something
  }
  public void operate(){
    this.method();
    super.operate();
  }
}
public class ConcreteDecorator2 extends Decorator{
  public ConcreteDecorator2(Component component){
    super(component);
  }
  //定义自己的修饰方法
  public void method(){
    //code to do something
  }
  public void operate(){
    this.method();
    super.operate();
  }
}
场景类
public class Client{
  public static void main(String[] args) {
    Component component = new ConcreteComponent();
    // 第一次修饰
    component = new ConcreteDecorator1(component);
    // 第二次修饰
    component = new ConcreteDecorator2(component);
    //运行结果
    component.operate();
  }
}

装饰模式的优缺点

装饰模式的优点
  • 装饰类和被装饰类可以独立发展,不会相互耦合。
    • Component类无需知道Decorator类,Decorator类是从外部来扩展Component类的功能,Decorator也不用知道具体的构件
  • 装饰模式是继承关系的一个替代方案
    • 装饰类Decorator不管装饰多少层,返回的对象还是Component,实现的还是is-a关系
  • 装饰模式可以动态地扩展一个实现类的功能,这不需要多说,装饰模式的定义就是如此
装饰模式的缺点
  • 多层的装饰是比较复杂的
  • 尽量减少装饰类的数量,一遍降低系统的复杂度
装饰模式的使用场景
  • 需要扩展一个类的功能,或给出一个类增加附加功能
  • 需要动态地给一个对象增加功能,这些功能可以再动态地撤销
  • 需要为一批的兄弟类进行改装或加装功能,当然是首选装饰模式
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,658评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,482评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,213评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,395评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,487评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,523评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,525评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,300评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,753评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,048评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,223评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,905评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,541评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,168评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,417评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,094评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,088评论 2 352

推荐阅读更多精彩内容

  • 模式动机 一般有两种方式可以实现给一个类或对象增加行为: 继承机制,使用继承机制是给现有类添加功能的一种有效途径,...
    lever_xu阅读 258评论 0 0
  • 需求 写一个给人模拟搭配不同服饰的程序,可以给人换各种各样的衣服裤子的形象。 初步实现 需求比较简单,直接上代码:...
    Mr丶sorrow阅读 754评论 0 2
  • 1 场景问题# 1.1 复杂的奖金计算## 考虑这样一个实际应用:就是如何实现灵活的奖金计算。 奖金计算是相对复杂...
    七寸知架构阅读 3,987评论 4 67
  • 原文地址:LoveDev 装饰模式(Decorator Pattern):也可以称为包装模式(Wrapper Pa...
    KevinLive阅读 618评论 1 2
  • 那是一个寻常的黄昏,舒胤将炎炎埋在了村子西边的柳树林里。 炎炎是条单身的老狗。 按人类的年龄来计算,大概已经很老很...
    啊煮到我的手辣_阅读 304评论 0 2