当系统需要新的功能的时候,一般都是向旧的类中添加新的代码。比如一个人这个类,需要新增穿衣的功能,那么就会在人这个类中去添加对应的穿衣方法代码,用来增强人这个类的行为功能。但是这样做,会因为新加入的字段、方法等逻辑代码,导致原有 人 这个类变得复杂,新加入的穿衣功能并非 人  这个类的核心功能,穿衣只是某种情况下的特定功能,会与类中人的核心功能【比如吃饭】代码糅杂在一起。

  装饰模式,把要装饰的功能(穿衣)放在一个单独的类中,并让这个类包含他需要装饰的对象(人),当需要执行特定穿衣功能的时候,客户端灵活去装饰主类对象。这样可以使得主类(人)代码简洁,不会变的越来越复杂,使得主类 的核心功能 与    不断增加的 特定功能 分开,也能去重相关装饰逻辑,做到复用。

下面写一个例子:

创建人 类作为主类:

public class Person {

    //人的 核心功能  吃喝拉撒
    public void eat() {
        System.out.println("人的核心功能:吃喝拉撒");
    }

    //穿衣功能  新增
    public void show() {
        System.out.println("====完成穿衣");
    }
}

创建装饰抽象类:

 * @description :装饰抽象类
 */
public abstract class Decorate extends Person {

    //Decorate extends Person
    //Person是被装饰者,  Decorate是装饰者,两者有共同的父类,这里简化了用继承来表示这种关系。目的是表明二者为同一类型,二者本身行为属性没有任何逻辑上的继承关系

    //被装饰对象   人      类
    private Person person;

    //装饰操作方法,为了让子类调用,不能为private
    public void decorate(Person person) {
        this.person = person;
    }

    //装饰行为效果
    @Override
    public void show() {
        if(person != null) {
            person.show();
        }
    }

}

创建三个装饰类,继承上面的装饰抽象类,来实现具体的装饰过程效果:

 * @description :女仆装装饰类
 */
public class Nvzhuang extends Decorate {

    @Override
    public void show() {
        System.out.println("穿女仆装");
        super.show();
    }

}

 * @description :裤子装饰类
 */
public class Kuzi extends Decorate {
    @Override
    public void show() {
        System.out.println("穿裤子");
        super.show();
    }

}

 * @description :马甲装饰类
 */
public class Majia extends Decorate {
    @Override
    public void show() {
        System.out.println("穿马甲");
        super.show();
    }

}

测试主类:

 */
public class Main {

    public static void main(String[] args) {

        Person xiaomingPerson = new Person();//创建被装饰对象,小明

        Nvzhuang nvzhuang  = new Nvzhuang();//女装 装饰类
        Majia majia  = new Majia();//马甲装饰类
        Kuzi kuzi = new Kuzi();//裤子装饰类

        //如何装饰?随便自定义顺序
        nvzhuang.decorate(xiaomingPerson);// 把下面定义的穿衣装饰顺序 套到  目标对象上
        majia.decorate(nvzhuang);//先穿马甲,再穿女装
        kuzi.decorate(majia);  //先穿裤子,再穿马甲
        kuzi.show();

    }

}

效果:

穿裤子
穿马甲
穿女仆装
====完成穿衣

注意: 代码中的装饰顺序是 女仆装、马甲,裤子, 运行效果却是 裤子、马甲、女仆装 ,顺序是反过来的,这其实就是在

nvzhuang.decorate(xiaomingPerson);// 把下面定义的穿衣装饰顺序 套到 目标对象上
majia.decorate(nvzhuang);//先穿马甲,再穿女装
kuzi.decorate(majia); //先穿裤子,再穿马甲

代码进行层层递进,最终到达裤子的时候,调用

 kuzi.show();开始进行装饰,裤子装饰完成后会调用上一层的装饰对象的装饰方法进行装饰。依次往回递进。

======================================

num05---装饰模式的更多相关文章

  1. 设计模式--装饰模式Decorate(结构型)

    一.装饰模式 动态地给一个对象添加额外的职责.就增加功能来说,装饰模式相比生成子类更为灵活.有时我们希望给某个对象而不是整个类添加一些功能. 二.UML图 1.Component(概念中提到的对象接口 ...

  2. Objective-C 装饰模式--简单介绍和使用

    装饰模式是在不必改变原类文件和使用继承的情况下,动态地扩展一个对象的功能.它是通过创建一个包装对象,也就是装饰来包裹真实的对象. 比如游戏机有一个GamePad类, 现在要增加一个作弊功能(例如100 ...

  3. C#设计模式系列:装饰模式(Decorator)

    1. 装饰模式简介 装饰模式动态地给一个对象添加额外的职责.例如一幅画有没有画框都可以挂在墙上,画就是被装饰者.但是通常都是有画框的.在挂在墙上之前,画可以被蒙上玻璃,装到框子里,所以在画上加一层画框 ...

  4. 装饰模式 - Decorator 和 外观模式 - Facade

    装饰模式 Decorator,不改变接口但动态给对象加入责任,所需功能按顺序串联起来控制,比生成子类灵活. 外观模式 Facade,让接口更简单.为子系统中的一组接口提供一个一致的界面. 参考:

  5. php实现设计模式之 装饰模式

    <?php /* * 装饰模式:在不必改变原类文件和使用继承的情况下,动态地扩展一个对象的功能.它是通过创建一个包装对象,也就是装饰来包裹真实的对象. * * 角色 * 抽象构件(Compone ...

  6. [工作中的设计模式]装饰模式decorator

    一.模式解析 装饰模式又名包装(Wrapper)模式.装饰模式以对客户端透明的方式扩展对象的功能,是继承关系的一个替代方案. 装饰模式的要点主要是: 1.需要对已有对象扩展新的功能,又不希望改变原有对 ...

  7. 基于东北F4的设计模式情景剧——第一幕 装饰模式(Decorator Pattern)

    第一场 难题未解 布景:铁岭,晴天,午后,风.在一幢还算气派的写字楼的三层外墙上,挂着一条红色横幅,上面用歪歪扭扭的毛笔字写着"东北F4软件外包工作室".大风中,那早已褪色的条幅剧 ...

  8. 基础笔记10(IO 1.7try-with-resource) 装饰模式

    1.读写的类型分为字节流和字符流,字节流一般是视频音频其他所有的类型都可以. (非文档文件使用字符流易造成未知编码(?)错误) InputStream OutputStream 抽象类 fileInp ...

  9. C++设计模式-Decorator装饰模式

    Decorator装饰模式作用:动态地给一个对象添加一些额外的职责,就增加功能来说,装饰模式比生成子类更为灵活. UML图如下: Component是定义一个对象接口,可以给这些对象动态地添加职责. ...

  10. 装饰模式(Decorator pattern)

    装饰模式(Decorator pattern): 又名包装模式(Wrapper pattern), 它以对客户端透明的方式扩展对象的功能,是继承关系的一个替代方案. 装饰模式以对客户透明的方式动态的给 ...

随机推荐

  1. 【一起学源码-微服务】Feign 源码一:源码初探,通过Demo Debug Feign源码

    前言 前情回顾 上一讲深入的讲解了Ribbon的初始化过程及Ribbon与Eureka的整合代码,与Eureka整合的类就是DiscoveryEnableNIWSServerList,同时在Dynam ...

  2. Arduino_URO端口与AtMega328p引脚对应图

    Arduino微控制器的数字端口和模拟端口与ATMEGA 328芯片引脚的对应关系图如下.标有0~13标号的引脚对应的是数字端口,在0~13前面有符号“~”的引脚对应的端口具有PWM输出功能.标有A0 ...

  3. Spring Boot 集成 Seata 解决分布式事务问题

    seata 简介 Seata 是 阿里巴巴2019年开源的分布式事务解决方案,致力于在微服务架构下提供高性能和简单易用的分布式事务服务.在 Seata 开源之前,Seata 对应的内部版本在阿里内部一 ...

  4. 【转】Zookeeper原理

    ZooKeeper是一个分布式的,开放源码的分布式应用程序协调服务,它包含一个简单的原语集,分布式应用程序可以基于它实现同步服务,配置维护和命名服务等.Zookeeper是hadoop的一个子项目,其 ...

  5. python之对象回收机制

    python中,当程序执行完毕之后,python的垃圾回收机制就会将所有对象回收,清除占用的内存 请看如下代码 class Parent(): def __init__(self,name): sel ...

  6. python命名空间(namespace)

    命名空间: 每一个作用域变量存储的位置,或者解释为 存储作用域中变量的字典. 作用: 获取想查看某个作用域中的变量名.变量值. 使用方法: locals()  #当前命名空间 1. 效果图: 2. 代 ...

  7. pip 下载源更换

    Python博大精深之处在于丰富的库,而目前最方便的下载库的方法无疑是pip.但是由于国内的网络环境导致,好多库下载是在太慢了,原因大家都懂得.. 一.临时修改 使用pip的时候加入参数 -i pip ...

  8. Spring Boot自动装配

    前言 一些朋友问我怎么读源码,这篇文章结合我看源码时候一些思路给大家聊聊,我主要从这三个方向出发: 确定目标,这个目标要是一个具体,不要一上来我要看懂Spring,这是不可能的,目标要这么来定,比如看 ...

  9. http请求头中的content-type属性

    在HTTP请求中,我们每天都在使用Content-Type来指定不同格式的请求信息,但是却很少有人去全面了解Content-Type中允许的值有多少,因此这里来了解一下Content-Type的可用值 ...

  10. es7中数组如何判断元素是否存在

    const arr = [1,2,3,4,5,6] console.log(arr.includes(4)) //true