函数应该做一件事,做好这件事,只做这一件事。

                    ——Clean Code

装饰模式,上代码:

先定义零件类:

 package cn.no3.decorator.template;

 public abstract class Component {

     public abstract void operation();
}

再定义具体零件类(即被装饰类)继承零件类:

 package cn.no3.decorator.template;

 public class ConcreteComponent extends Component{

     @Override
public void operation() {
// TODO Auto-generated method stub
System.out.println("对一个Object进行装饰:");
} }

然后定义装饰类,继承零件类:

 package cn.no3.decorator.template;

 public class Decorator extends Component {

     protected Component component;

     public void setComponent(Component component) {
this.component = component;
} @Override
public void operation() {
//调用的是传入component的operation方法
if (component != null) {
component.operation();
} } }

定义装饰类的子类:

 package cn.no3.decorator.template;

 public class ConcreteDecoratorA extends Decorator {

     private String addedField;

     @Override
public void operation() {
// TODO Auto-generated method stub
super.operation();//第一行调用super,则从前往后包装.反之则从后往前包装
addedField = "新的装饰物A";
System.out.println("装饰了:"+addedField);
}
} public class ConcreteDecoratorB extends Decorator { @Override
public void operation() {
// TODO Auto-generated method stub
super.operation();
AddedBehavior();
} private void AddedBehavior() {
// TODO Auto-generated method stub
System.out.println("完成装饰,最后照个相");
} }

测试类测试:

 package cn.no3.decorator.template;

 public class _Test {

     public static void main(String[] args) {
//新建装饰对象
ConcreteComponent concreteComponent = new ConcreteComponent();
ConcreteDecoratorA concreteDecoratorA = new ConcreteDecoratorA();
ConcreteDecoratorB concreteDecoratorB = new ConcreteDecoratorB();
//逐层包装
//如果注掉下句,还是先调用A的super,发现没传component,于是A的super什么都不做
concreteDecoratorA.setComponent(concreteComponent);//A先调用cC,cC方法不再调用super.
concreteDecoratorB.setComponent(concreteDecoratorA);//B先调用A
//输出最终结果
concreteDecoratorB.operation();
}
}

把这个例子中的Component类换成接口也是一样的:

举个例子:

需求:

定义一个角色,让他实现IWearable接口,

再定义装备实现IWearable接口

人物装备装备.

上代码:

 package cn.no3.decorator.instance;

 public abstract interface IWearable {

     public abstract void operation();
}
 package cn.no3.decorator.instance;

 public class Clothes implements IWearable {

     protected IWearable component;

     public void setComponent(IWearable component) {
this.component = component;
} @Override
public void operation() {
if (component != null) {
component.operation();
} } }
 package cn.no3.decorator.instance;

 //即使只有一双鞋也要定义父类,否则无法通过对象的super.operation对装饰进行传递
public class Shoes implements IWearable { protected IWearable component; public void setComponent(IWearable component) {
this.component = component;
} @Override
public void operation() {
if (component != null) {
component.operation();
}
} }
 package cn.no3.decorator.instance;

 public class Weapons implements IWearable {

     protected IWearable component;

     public void setComponent(IWearable component) {
this.component = component;
} @Override
public void operation() {
if (component != null) {
component.operation();
} } }

他们的子类都是重写父类operation()方法,最先调用(或者最后调用),super.operation();此处省略若干行

 package cn.no3.decorator.instance;

 public class WeaponA extends Weapons {

     @Override
public void operation() {
// TODO Auto-generated method stub
super.operation();
System.out.println("装备了:霜之哀伤");
}
}

角色类

 package cn.no3.decorator.instance;

 public class Character implements IWearable{

     @Override
public void operation() {
// TODO Auto-generated method stub
System.out.println("对一个人物进行装备:");
} }

测试类:

 package cn.no3.decorator.instance;

 public class _Test {

     public static void main(String[] args) {
//新建装备对象
Character concreteComponent = new Character();
ClotheA fantasyKiller = new ClotheA();
ClotheB bravingArmor = new ClotheB();
WeaponA frostmourne = new WeaponA();
WeaponB DoubleRoses = new WeaponB();
ShoesA shoes = new ShoesA();
//逐层包装
fantasyKiller.setComponent(concreteComponent);
bravingArmor.setComponent(fantasyKiller);
frostmourne.setComponent(bravingArmor);
DoubleRoses.setComponent(frostmourne);
shoes.setComponent(DoubleRoses);//shoes.setComponent(shoes)会陷入死循环
//输出最终结果
shoes.operation();
}
}

装饰模式是为已有功能动态地添加更多功能的一种方式

装饰模式把每个要装饰的功能放在单独的类中,并让这个类包装它所需要装饰的对象。因此,当需要执行特殊行为时,客户代码就可以在运行时根据需要有选择地、按顺序地使用装饰功能包装对象了。

装饰功能的核心是:把类中的装饰功能从类中搬出去。

Java学习笔记——设计模式之三.装饰模式的更多相关文章

  1. Java学习笔记——设计模式之五.工厂方法

    水边一只青蛙在笑 --石头和水 工厂方法模式(Factory Method),定义了一个用于创建对象的接口,让实现类决定实例化哪一个类.工厂方法使一个类的实例化延迟到其子类. 这里在简单和工厂的基础上 ...

  2. Java学习笔记——设计模式之二.策略模式

    明确是王道 --Clean Code 先定义策略类 package cn.no2.strategy; public abstract class Strategy { //省略属性 //算法方法 pu ...

  3. Java学习笔记——设计模式之一.简单工厂

    蜀道之难.难于上青天,侧身西望长咨嗟 --蜀道难 设计模式第一篇,简单工厂. 定义Operation类 package cn.no1.simplefactory; public abstract cl ...

  4. Java学习笔记——设计模式之十.观察者模式

     观察者模式(Observer),定义了一种一对多的依赖关系,让多个观察者对象同时监听某一个主题对象.这个主题对象在状态发生变化时,会通知所有观察者对象,使他们能够自动更新自己. Subject类: ...

  5. Java学习笔记——设计模式之九.建造者模式

     建造者模式(Builder),将一个复杂对象的构建与它的表示分离,使得同样的构建过程可以创建不同的表示. Product类: package cn.happy.design_pattern._09b ...

  6. Java学习笔记-设计模式

    常见设计模式有23种,设计模式是解决某一类问题最行之有效的方法 单例设计模式 解决一个类在内存中只存在一个对象 思想 为了避免其他程序过多建立该类对象.先禁止其他程序建立该类对象 还为了让其他程序可以 ...

  7. Java学习笔记——设计模式之四.代理模式

    To be, or not to be: that is the question. --<哈姆雷特> 代理模式(Proxy),为其他对象提供一种代理以控制对这个对象的访问. 上代码: p ...

  8. Java学习笔记——设计模式之八.外观模式

    外观模式(Facade),为子系统中的一组接口提供一个一致的界面,此模式定义了一个高层接口,这个接口使得这一子系统更加容易使用. 子系统: package cn.happy.design_patter ...

  9. Java学习笔记——设计模式之七.模板方法模式

    模板方法模式(TemplateMethod),定义一个操作中的算法的骨架,而将一些步骤延迟到子类中.模板方法使得子类可以不改变一个算法的结构即可重定义该算法的某些特定步骤. 结构图: 代码: 算法骨架 ...

随机推荐

  1. 一种类似Retrofit声明接口即可实现调用的WebApi客户端框架

    为.Net出力 java有okhttp,还在okhttp这上搞了一个retrofit,.net有HttpClient,但目前我没有发现有类似的retrofit框架.最近在搞mqtt的webApi封装, ...

  2. java 接口默认修饰符问题

    package test; public interface InterfaceTest { void todo();} /** * 以下是反编译的内容,接口里的方法默认都是public的.abstr ...

  3. 【C++】模拟实现auto_ptr

    看了<Effctive C++>,里面提到用对象去管理资源,可以有效防止内存泄漏. 结合auto_ptr特性,稍微思考了一下,实现了一个简单的auto_ptr (因为代码量小,就不分文件了 ...

  4. 自动化利器-RPM自定义打包

    1.Rpm打包程序 1.1为什么要使用rpm打包 1.编译安装软件,优点是可以定制化安装目录.按需开启功能等,缺点是需要查找并实验出适合的编译参数,诸如MySQL之类的软件编译耗时过长. 2.yum安 ...

  5. Python之路-awk文本处理

    作业一:整理博客,内容包含awk.变量.运算符.if多分支 一.awk 1.awk是一个优秀的文本处理工具,多用来处理含有特殊分隔符的内容 常见用法 awk -F:  {print $1,$4} 作业 ...

  6. Html5 Canvas笔记(1)-CanvasAppTemplate代码

    学了一段时间的Html5 Canvas,现想一段一段的将学习笔记整理出来放上来,先整理一段Canvas的模版文件代码,以后建立Canvas程序就不用重新写这些代码了,当然最好是将这个Html代码保存到 ...

  7. [原]docker 操作记录

    开启新容器 docker run --name 容器名字 -ti[d] 镜像 初始化命令(需要是阻塞的) 额外参数 -v 本地目录:容器目录[:ro] 映射本地路径和容器路径(时区同步.数据库dock ...

  8. 如何在 FineUIMvc 中引用第三方 JavaScript 库

    声明:FineUIMvc(基础版)是免费软件,本系列文章适用于基础版. 引入第三方颜色选择器 在 FineUIMvc 中使用第三方 JavaScript 遵循一定的约定,也非常简单. 下面以官网示例为 ...

  9. C++模板--实现容器适配器

    STL源码初接触 STL = Standard Template Library,直译过来是:标准模板库,是惠普实验室开发的一系列软件的统称.从根本上说,STL是一些"容器"的集合 ...

  10. 判断iframe页面是否加载完成

        if (frames("appIframe").document.readyState !="complete") {          alert(& ...