要实现装饰者模式,注意一下几点内容:
1.装饰者类要实现真实类同样的接口
2.装饰者类内有一个真实对象的引用(可以通过装饰者类的构造器传入)
3.装饰类对象在主类中接受请求,将请求发送给真实的对象(相当于已经将引用传递到了装饰类的真实对象)
4.装饰者可以在传入真实对象后,增加一些附加功能(因为装饰对象和真实对象都有同样的方法,装饰对象可以添加一定操作在调用真实对象的方法,或者先调用真实对象的方法,再添加自己的方法)
5.不用继承,
 
先用实例说话,最后再具体装饰者模式
 
假设要制造添加甜蜜素和着色剂的馒头:
1.需要生产一个正常馒头
2.为节省成本(不使用玉米面),使用染色剂加入到正常馒头中
3.和面,最后生产出染色馒头
 
一..先实现做面包的接口
    IBread接口包括准备材料,和面,蒸馒头,加工馒头(即调用前面三个步骤)
 package 装饰模式;

 public interface IBread {

     public void prepair();

     public void kneadFlour();

     public void steamed();

     public void process();
}
 
 
二.制作正常馒头
 package 装饰模式;

 public class NormalBread implements IBread{

     @Override
public void prepair() { System.out.println("准备面粉,水以及发酵粉...");
} @Override
public void kneadFlour() { System.out.println("和面...");
} @Override
public void steamed() { System.out.println("蒸馒头...香喷喷的馒头出炉了");
} @Override
public void process() { prepair();
kneadFlour();
steamed();
} }
 
三.定义出制作面包的抽象类
    抽象类实现了IBread这个制作面包的接口,同时包含IBread接口的实例
    对应上述的第2个注意点:装饰者类内有一个真实对象的引用
 package 装饰模式;

 public abstract class AbstractBread implements IBread {

     private final IBread bread;

     public AbstractBread(IBread bread) {
super();
this.bread = bread;
}
@Override
public void prepair() {
this.bread.prepair();
}
@Override
public void kneadFlour() {
this.bread.kneadFlour();
}
@Override
public void steamed() {
this.bread.steamed();
} @Override
public void process() {
prepair();
kneadFlour();
steamed();
} }
 
 
 
四.生产有着色剂的"玉米馒头"
    继承AbstarctBread类,所以可以有选择的覆盖正常生产馒头的方法,并添加原有方法原来的信息,同时也可以添加自己的方法
    装饰者模式中这里最关键,
     对应上述的第1个注意点:装饰者类要实现真实类同样的接口
 package 装饰模式;

 public class CornDecorator extends AbstractBread{

     public CornDecorator(IBread bread) {

         super(bread);
} public void paint(){ System.out.println("添加柠檬黄的着色剂");
}
@Override
public void kneadFlour() {
//添加着色剂后和面
this.paint();
super.kneadFlour();
} }
 
 
五.生产有甜蜜素的"甜馒头"
    实现与第四部一样
 package 装饰模式;

 public class SweetDecorator extends AbstractBread {

     public SweetDecorator(IBread bread) {

         super(bread);
} public void paint(){ System.out.println("添加甜蜜素...");
} @Override
public void kneadFlour() {
//添加甜蜜素后和面
this.paint();
super.kneadFlour();
} }
 
六.开始制作添加甜蜜素和着色剂的馒头
 package 装饰模式;

 public class Client {

     public static void main(String[] args) {

         System.out.println("=======开始装饰馒头");
IBread normalBread = new NormalBread();
normalBread = new SweetDecorator(normalBread);
normalBread = new CornDecorator(normalBread);
normalBread.process();
System.out.println("=======装饰馒头结束");
}
}
 
七.输出
 =======开始装饰馒头
准备面粉,水以及发酵粉...
添加柠檬黄的着色剂
添加甜蜜素...
和面...
蒸馒头...香喷喷的馒头出炉了
=======装饰馒头结束
装饰者模式中的4个角色
(1)被装饰者抽象Component:是一个接口或者抽象类,定义最核心的对象,这个类是装饰者的基类,例如IBread接口
(2)被装饰者具体实现ConcreteComponent:这是Component接口或抽象类的实现,例如本例中的NormalBread
(3)装饰者Decorator:一般是抽象类,实现Component,它里面必然有一个指向Component的引用,例如本例中AbstractBread
(4)装饰者实现ConcreteDecorator1和ConcreteDecorator2:用来装饰最基本的类,如本例中的CornDecorator,
 
JDK中的装饰者模式
 java.io中很多使用了装饰者模式
举个例子:FilterInputStream继承(实现)了InputStream,同时,BufferedInputStream继承了FilterInputStream,
1,被装饰者抽象组件:即最顶层的基类InputStream
2.被装饰者具体实现ConcreteComponent:FileInputStream和FileOutputStream就是它的实现
3.装饰者Decorator:FilterInputStream中有一个InputStream的实例和构造方法传入InputStream对象
    protected volatile InputStream in;  
    protected FilterInputStream(InputStream in) {
        this.in = in;

}

4.装饰者实现:在   BufferedInputStream  中有构造方法传入InputStream对象,实现了装饰
    public BufferedInputStream(InputStream in, int size) {
        super(in);
        if (size <= 0) {
            throw new IllegalArgumentException("Buffer size <= 0");
        }
        buf = new byte[size];

}

这个构造方法,对比上面的做面包流程,可以惊奇的发现是一模一样的. (可以将)
1.InputStream-->IBread
    (这里就是InputStream,没什么好说的)
2.FileInputStream-->NormalBread
3.FilterInputStream-->AbstractBread
  (实现Component,这里是InputStram.它里面必然有一个指向Component的引用,这个引用就是InputStream的实例)
4. BufferedInputStream  --> CornDecorator
    (装饰者实现,:用来装饰最基本的类,对I传入的nputStream进行了装饰)
  (BufferedInputStream 的super(in)就像CornDecorator中的kneadFlour()方法,也有super.kneadFlour();,只是这里IO流中用在了构造方法 )
这就是JDK中的装饰者模式
 
 
 
不用继承方式实现装饰者模式的原因(以此例为说明对象,)
1.如果只是单独的添加色素或者甜蜜素确实是可以做到的,只需要将CornDecorator继承NormalBread 和SweetDecorator 继承NormalBread ,这样也能够覆盖正常制作面包的流程,添加附加的功能实现单独制作"玉米馒头"和"甜馒头".
2.如此一来,如果我们要制作甜玉米馒头(这里加点先添加甜色素,再添加玉米色素),只需要先SweetDecorator继承 NormalBread,然后CornSweetDecorator 再用继承 CornDecorator ,这样似乎是没有问题的.
2.但是想想以下的情况,如果我们希望能够在添加在添加甜色素和玉米色素中间还要加入洋葱,这要怎么做,难道说又用先SweetDecorator继承 NormalBread,然后OnionSweetDecorator继承SweetDecorator,最后再用CornOnionSweetDecorator 继承OnionSweetDecorator??
显然是不可能的,这样会导致原来代码的复用性低,而且形成了冗余的继承体系
4.使用上述实例的方法完全克服了这个问题,要实现添加洋葱,只需要实现和SweetDecorator 类似的步骤即可,最后在Client类中传入就可以实现这个功能
 
 
使用场合
1.需要为某个现有对象添加一个新的功能或职责时,可以考虑使用装饰者模式
2.某个对象的职责经常发生变化或经常需要动态添加职责,避免为了适应这种变化造成的继承扩展方式
 
 
 

Java设计模式之装饰者模式的更多相关文章

  1. Java 设计模式泛谈&装饰者模式和单例模式

    设计模式(Design Pattern) 1.是一套被反复使用.多人知晓的,经过分类编目 的 代码设计经验总结.使用设计模式是为了可重用代码,让代码更容易维护以及扩展. 2.简单的讲:所谓模式就是得到 ...

  2. Java设计模式 - - 单例模式 装饰者模式

    Java设计模式 单例模式 装饰者模式 作者 : Stanley 罗昊 [转载请注明出处和署名,谢谢!] 静态代理模式:https://www.cnblogs.com/StanleyBlogs/p/1 ...

  3. Java设计模式系列-装饰器模式

    原创文章,转载请标注出处:<Java设计模式系列-装饰器模式> 一.概述 装饰器模式作用是针对目标方法进行增强,提供新的功能或者额外的功能. 不同于适配器模式和桥接模式,装饰器模式涉及的是 ...

  4. 23种java设计模式之装饰者模式及动态代理

    设计模式不管对于何种语言都是存在的,这里介绍的是java的模式 装饰者模式是在二次开发中应用比较多的一款模式,当然了用反射也是可以实现的,今天介绍的是装饰模式,有兴趣的朋友可以自己去了解一下反射是怎么 ...

  5. java设计模式—Decorator装饰者模式

    一.装饰者模式 1.定义及作用 该模式以对客户端透明的方式扩展对象的功能. 2.涉及角色      抽象构件角色:定义一个抽象接口,来规范准备附加功能的类. 具体构件角色:将要被附加功能的类,实现抽象 ...

  6. java设计模式之七装饰器模式(Decorator)

    顾名思义,装饰模式就是给一个对象增加一些新的功能,而且是动态的,要求装饰对象和被装饰对象实现同一个接口,装饰对象持有被装饰对象的实例,关系图如下: Source类是被装饰类,Decorator类是一个 ...

  7. java设计模式之 装饰器模式

    装饰器模式 装饰器模式(Decorator Pattern)允许向一个现有的对象添加新的功能,同时又不改变其结构. 这种类型的设计模式属于结构型模式,它是作为现有的类的一个包装. 这种模式创建了一个装 ...

  8. Java设计模式之装饰器模式

    1.装饰器模式的定义(保持接口,扩展功能) Decorate装饰器,顾名思义,就是动态的给一个对象添加一些额外的职责,就好比对房子进行装修一样. 2.装饰器模式的特征 具有一个装饰对象. 必须拥有与被 ...

  9. java设计模式之装饰者模式学习

    装饰者模式 Decorator模式(别名Wrapper):动态将职责附加到对象上,若要扩展功能,装饰者提供了比继承更具弹性的代替方案. 装饰者与被装饰者拥有共同的超类,继承的目的是继承类型,而不是行为 ...

随机推荐

  1. A*算法进入

    作者文章链接:http://www.policyalmanac.org/games/aStarTutorial.htm 启示式搜索:启示式搜索就是在状态空间中的搜索对每个搜索的位置进行评估,得到最好的 ...

  2. 高榕资本宾悦:未使用的企业家Testin云测试服务类故障

    高榕资本岳斌:创业者未使用Testin云測试服务属不合格 2014/10/09 · Testin · 开发人员訪谈 Testin云測与工信部等联合承办的ICT中国.2014高层论坛之移动开发人员分论坛 ...

  3. Android-管理Activity生命周期 -开始一个Activity

    很多程序都是从main()方法开始启动的,和其他程序不同,android是在activity生命周期的特定状态的特定回调方法中初始化代码的.activity启动和销毁的时候都用很多回调方法. 这里将要 ...

  4. UIStepper使用的具体解释的控制

    UIStepper控件类似于UISlider控件,但它有"+"和"-"两个button,单击当中一个可使属性value值递增或递减. 如声音.速度.图片等的大小 ...

  5. 自己动手写一个编译器Tiny语言解析器实现

    然后,上一篇文章简介Tiny词法分析,实现语言.本文将介绍Tiny的语法分析器的实现. 1 Tiny语言的语法 下图是Tiny在BNF中的文法. 文法的定义能够看出.INNY语言有以下特点: 1 程序 ...

  6. php 无错误提示 的解决方法

    问:我在win7安装了PHP,浏览器是IE9.我代码写错了,浏览器一点错误提示都没有,一片空白.如果写对了,就能正常运行显示出来.请问这是怎么回事,应该怎么弄?你们两个的方法都试过,但都没有提示(注: ...

  7. DBA工具——DMV——如何知道TSQL语句已运行了多久

    原文:DBA工具--DMV--如何知道TSQL语句已运行了多久 DBA通常想知道正在运行的语句已经执行了多久了?可以使用Sqlserver profiler来捕获语句的开始时间,和现有时间比较,但是在 ...

  8. 【白注意】Windows XP 大胆拥抱Linux在系统中所遇到的问题

    Windows XP至4月8日本将不再是微软官方技术支持.虽然仍可以继续使用,但他们大部分的风险.可一不留神被黑客攻击.似向下,Linux这也是一个不错的选择. 或许很多文章开始教你如何迁移,您,迁移 ...

  9. Android与服务器端数据交互(转)

    上一节中我们通过http协议,采用HttpClient向服务器端action请求数据.当然调用服务器端方法获取数据并不止这一种.WebService也可以为我们提供所需数据,那么什么是webServi ...

  10. Matlab中调用第三方Java代码

    搞了一天,才算搞定. 第一步:定位Matlab中Java环境的ext目录 新建一个M script文件,或者直接在Matlab的交互式命令行中输入: disp(java.lang.System.get ...