大话设计模式笔记(四)の装饰模式

举个栗子

问题描述

可以给人搭配嘻哈服或白领装的程序。

简单实现

代码

/**
 * 人类
 * Created by callmeDevil on 2019/6/23.
 */
public class Person {

    private String name;

    public Person(String name) {
        this.name = name;
    }

    public void wearTShirts(){
        System.out.print("大T恤 ");
    }

    public void wearBigTrouser(){
        System.out.print("垮裤 ");
    }

    public void wearSneakers(){
        System.out.print("破球鞋 ");
    }

    public void wearSuit(){
        System.out.print("西装 ");
    }

    public void wearTie(){
        System.out.print("领带 ");
    }

    public void wearLeatherShoes(){
        System.out.print("皮鞋 ");
    }

    public void show(){
        System.out.println("装扮的" + name);
    }

}
/**
 * 装扮测试类
 * Created by callmeDevil on 2019/6/23.
 */
public class Test {

    public static void main(String[] args) {
        Person devil = new Person("Devil");

        System.out.println("第一种装扮:");
        devil.wearTShirts();
        devil.wearBigTrouser();
        devil.wearSneakers();
        devil.show();

        System.out.println("\n第二种装扮:");
        devil.wearSuit();
        devil.wearTie();
        devil.wearLeatherShoes();
        devil.show();
    }

}

测试结果

第一种装扮:
大T恤 垮裤 破球鞋 装扮的Devil

第二种装扮:
西装 领带 皮鞋 装扮的Devil

存在缺陷

如果需要增加“超人”装扮,会导致需要修改“Person”类,违背了开放-封闭原则

简单实现进化版

代码

/**
 * 人类
 * Created by callmeDevil on 2019/6/23.
 */
public class Person {

    private String name;

    public Person(String name){
        this.name = name;
    }

    public void show(){
        System.out.print("装扮的" + name);
    }

}
/**
 * 服装抽象类
 * Created by callmeDevil on 2019/6/23.
 */
public abstract class Finery {

    public abstract void show();

}
/**
 * T恤 类
 * Created by callmeDevil on 2019/6/23.
 */
public class TShirts extends Finery {

    @Override
    public void show() {
        System.out.print("大T恤 ");
    }

}
/**
 * 垮裤 类
 * Created by callmeDevil on 2019/6/23.
 */
public class BigTrouser extends Finery {

    @Override
    public void show() {
        System.out.print("垮裤 ");
    }

}

// 其余子类相似,此处省略
/**
 * 装饰升级版测试
 * Created by callmeDevil on 2019/6/23.
 */
public class Test {

    public static void main(String[] args) {
        Person devil = new Person("Devil");

        System.out.println("第一种装扮:");
        Finery tShirts = new TShirts();
        Finery bigTrouser = new BigTrouser();
        Finery sneakers = new Sneakers();
        tShirts.show();
        bigTrouser.show();
        sneakers.show();
        devil.show();

        System.out.println("\n第二种装扮:");
        Finery suit = new Suit();
        Finery tie = new Tie();
        Finery leatherShoes = new LeatherShoes();
        suit.show();
        tie.show();
        leatherShoes.show();
        devil.show();
    }

}

测试结果

第一种装扮:
大T恤 垮裤 破球鞋  装扮的Devil

第二种装扮:
西装 领带 皮鞋 装扮的Devil

存在问题

现在如果要加超人装扮,只要增加子类就可以了,但是这么做虽然把“服装”类和“人”类分离开了,仍然是存在问题的。把“大T恤”、“垮裤”、“破球鞋”和“装扮的Devil”一个词一个词显示出来,就好比:你光着身子,当着大家的面,先穿T恤,再穿裤子,再穿鞋,仿佛在跳穿衣舞。。。因此需要一个房间(组合类)来换衣服,同时这个穿的顺序对每个人来说是不固定的,有的人喜欢先穿裤子,再穿鞋,最后穿T恤。。只需要把所需的功能按正确的顺序串联起来进行控制即可。

装饰模式

定义

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

UML图

  • Component是定义一个对象接口,可以给这些对象动态地添加职责。
  • ConcreteComponent是定义了一个具体的对象,也可以给这个对象添加一些职责。
  • Decorator,装饰抽象类,继承了Component,从外类来扩展Component类功能,但对于Component来说,是无需知道Decorator的存在的。
  • 至于ConcreteDecorator就是具体的装饰对象,起到了给Component添加职责的功能。

简单实现究极进化->装饰模式实现

代码

/**
 * 究极进化人类(ConcreteComponent)
 * Created by callmeDevil on 2019/6/23.
 */
public class Person {

    private String name;

    public Person(){}

    public Person(String name){
        this.name = name;
    }

    public void show(){
        System.out.println("装扮的" + name);
    }

}
/**
 * 究极进化 服饰类(Decorator)
 * Created by callmeDevil on 2019/6/23.
 */
public abstract class Finery extends Person{

    protected Person component;

    /**
     * 装扮
     * @param component
     */
    public void decorate(Person component) {
        this.component = component;
    }

    @Override
    public void show() {
        if (component != null) {
            component.show();
        }
    }

}
/**
 * 究极进化 T恤(ConcreteDecorator)
 * Created by callmeDevil on 2019/6/23.
 */
public class TShirts extends Finery{

    @Override
    public void show() {
        System.out.print("大T恤 ");
        super.show();
    }

}
/**
 * 究极进化 垮裤(ConcreteDecorator)
 * Created by callmeDevil on 2019/6/23.
 */
public class BigTrouser extends Finery {

    @Override
    public void show() {
        System.out.print("垮裤 ");
        super.show();
    }

}

// 其余子类相似,此处省略
/**
 * 装饰模式测试
 * Created by callmeDevil on 2019/6/23.
 */
public class Test {

    public static void main(String[] args) {
        Person devil = new Person("Devil");

        System.out.println("第一种装扮:");
        Sneakers sneakers = new Sneakers();
        BigTrouser bigTrouser = new BigTrouser();
        TShirts tShirts = new TShirts();
        // 装饰
        sneakers.decorate(devil);
        bigTrouser.decorate(sneakers);
        tShirts.decorate(bigTrouser);
        tShirts.show();

        System.out.println("\n第二种装扮:");
        LeatherShoes leatherShoes = new LeatherShoes();
        Tie tie = new Tie();
        Suit suit = new Suit();
        // 装饰
        leatherShoes.decorate(devil);
        tie.decorate(leatherShoes);
        suit.decorate(tie);
        suit.show();
    }

}

测试结果

第一种装扮:
大T恤 垮裤 破球鞋  装扮的Devil

第二种装扮:
西装 领带 皮鞋 装扮的Devil

总结

  • 装饰模式是利用setComponent来对对象进行包装的,这样每个装饰对象的实现就和如何使用这个对象分离开了,每个装饰对象只关心自己的功能,不需要关心如何被添加到对象链当中。
  • 如果只有一个ConcreteComponent类而没有抽象的Component类,那么Decorator类可以是ConcreteComponent的一个子类。同样道理,如果只有一个ConcreteDecorator类,那么就没有必要建立一个单独的Decorator类,而可以把DecoratorConcreteDecorator的责任合并成一个类。
  • 装饰模式是为已有功能动态地添加更多功能的一种方式
  • 不使用装饰模式,且系统需要新功能的时候,是向旧的类中添加新的代码,这些新加的代码通常装饰了原有类的核心职责或主要行为。在主类中加入了新的字段、方法和逻辑,会增加主类的复杂度,而这些新加入的东西仅仅是为了满足一些只在某种特定情况下才会执行的特殊需求。装饰模式提供了非常好的解决方案,它把每个要装饰的功能放在单独的类中,并让这个类包装它所要装饰的对象,因此,当需要执行特殊行为时,客户代码就可以在运行时根据需要有选择地、按顺序地使用装饰功能包装对象。
  • 装饰模式的优点是把类中装饰功能从类中搬移去除,这样可以简化原有的类。这么做最大的好处就是有效地把类的核心职责和装饰功能区分开了,而且可以去除相关类中重复的装饰逻辑。
(0)

相关推荐

  • 大话设计-装饰模式

    装饰模式的核心:被装饰对象(最原始),装饰对象(带有装饰功能的被装饰对象). 假设有一个被装饰对象叫"ConcreteComponent ",两个装饰对象叫"Concre ...

  • 大话设计模式笔记(七)の原型模式

    举个栗子 问题描述 要求有一个简历类,必须要有姓名,可以设置性别和年龄,可以设置工作经历,最终需要三份简历. 简单实现 简历类 /** * 简历类 * Created by callmeDevil o ...

  • 详解JAVA面向对象的设计模式 (七)、装饰模式

    装饰模式 Decorator 装饰模式比较简单,我就不单独写实现例子了.参考设计图去实现不是什么问题.建议可以写一写找找感觉. 在现实生活中,常常需要对现有产品增加新的功能或美化其外观,如房子装修.相 ...

  • DercoratorPattern装饰者模式

    装饰者模式 1,定义 动态的给一个对象添加一些额外的职责. 装饰者模式通常有4个角色,就增加功能来说,装饰者模式比生成子类更加的灵活. Component:抽象构件.通常是一个接口或者抽象类,定义最核 ...

  • 大话设计模式笔记(五)の代理模式

    举个栗子 故事是这样的... 一个小伙子喜欢上了隔壁班的一个妹子,但是又不认识,也害羞不好意思主动去说话,于是拜托了同样在这个班的一个朋友去传递自己想要送的礼物... 代码实现 该模式就不上什么简单实 ...

  • 设计模式-结构型-装饰者模式

    装饰者模式(wrapper): 允许向一个现有的对象添加新的功能,同时又不改变其结构.装饰器模式是一种用于代替继承的技术,无需通过继承增加子类就能扩展对象的新功能.使用对象的关联关系代替继承关系,更加 ...

  • 大话设计模式笔记(十八)の单例模式

    单例模式 定义 保证一个类仅有一个实例,并提供一个访问它的全局访问点. 通常我们可以让一个全局变量使得一个对象被访问,但它不能防止你实例化多个对象.一个最好的办法就是,让类自身负责保存它的唯一实例.这 ...

  • 大话设计模式笔记(十七)の迭代器模式

    迭代器模式 定义 提供一种方法顺序访问一个聚合对象中各个元素,而又不暴露该对象的内部表示. 什么时候用? 当你需要访问一个聚集对象,而且不管这些对象是什么都需要遍历的时候,你就应该考虑用迭代器模式. ...

  • 大话设计模式笔记(十三)の状态模式

    举个栗子 问题描述 上班的日子,上午状态好,中午想睡觉,下午渐恢复,加班苦煎熬.根据时间的不同体现不同的工作状态. 简单实现 Work /** * 工作类 * Created by callmeDev ...

  • 大话设计模式笔记(六)の工厂方法模式

    栗子回顾 简单工厂模式: https://www.cnblogs.com/call-me-devil/p/10926633.html 运算类使用工厂方法模式实现 UML图 代码实现 工厂接口 /** ...

  • 大话设计模式笔记(二)の策略模式

    举个栗子 问题描述 商场收银软件,营业员根据客户所购买的商品单价和数量,向客户收费. 简单实现 /** * 普通实现 * Created by callmeDevil on 2019/6/1. */ ...

  • 大话设计模式笔记(一)の简单工厂模式

    概要 一个好的程序猿/媛敲出来的代码应该是可维护.可复用.可扩展的,也就是具有较好的灵活性. 为了达到以上目的,在还没敲代码之前,需要事先考虑通过何种方式能够使自己的程序的耦合度降低,最基本的便是面向 ...

  • [读书笔记] 《大话设计模式》

    作者:程杰 前言 成为 诗人 后 可能 不需要 刻意 地 按照 某种 模式 去 创作, 但 成为 诗人 前 他们 一定 是 认真 地 研究 过 成百上千 的 唐诗 宋词. 古今 名句. 第 1 章 代 ...