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

                    ——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. c++学习笔记之封装篇(上)

    title: c++学习笔记之封装篇(上) date: 2017-03-12 18:59:01 tags: [c++,c,封装,类] categories: [学习,程序员,c/c++] --- 一. ...

  2. (转)经验分享:CSS浮动(float,clear)通俗讲解

    很早以前就接触过CSS,但对于浮动始终非常迷惑,可能是自身理解能力差,也可能是没能遇到一篇通俗的教程. 前些天小菜终于搞懂了浮动的基本原理,迫不及待的分享给大家. 写在前面的话: 由于CSS内容比较多 ...

  3. css删除线,下划线等

    <style> .p1 { text-decoration:overline; //上划线 } .p2 { text-decoration:line-through; //删除线 }  . ...

  4. 20155304 2016-2017-2 《Java程序设计》第五周学习总结

    20155304 2016-2017-2 <Java程序设计>第五周学习总结 教材学习内容总结 第八章 try catch JVM会先尝试执行try区块中的内容,若发生错误且与catch后 ...

  5. 腾讯云数据库团队:PostgreSQL TOAST技术理解

    作者介绍:胡彬 腾讯云高级工程师 TOAST是"The Oversized-Attribute Storage Technique"的缩写,主要用于存储一个大字段的值.要理解TOA ...

  6. css2.1实现圆角边框

    虽然css3的border-radius实现圆角很简单,但是我还是认为css2.1中好多技术还是很值得学习的,我也是后来才知道这就是传说中的滑动门技术.脑洞大开啊 附上demo <!DOCTYP ...

  7. React-Native 之 项目实战(三)

    前言 本文有配套视频,可以酌情观看. 文中内容因各人理解不同,可能会有所偏差,欢迎朋友们联系我. 文中所有内容仅供学习交流之用,不可用于商业用途,如因此引起的相关法律法规责任,与我无关. 如文中内容对 ...

  8. GWT开端

    这篇文章是转载的,原地址:GWT开端 以前的基于GWT的项目中广泛使用的还是gwt-windows-1.5.3这个版本的. 1.下载地址:http://code.google.com/webtoolk ...

  9. 【Electron】Electron开发入门(一):开发环境搭建

    刚接触Electron+js开发PC端桌面应用程序的时候,简直一头雾水,搜了网上很多教程,有的要么讲的零零碎碎,要么就是版本太低,很多API语法都不能用了:现在我把一些有用的教程归纳一下,并把目前最新 ...

  10. 4.Maven仓库

    1. 何为Maven仓库 Maven仓库就是统一存放所有依赖的地方,其他所有项目都可以在仓库里通过坐标找到所需要的依赖. 2. 仓库的布局 任何一个构件都有其唯一的坐标,根据这个坐标可以定义其在仓库中 ...