Markdown版本笔记 我的GitHub首页 我的博客 我的微信 我的邮箱
MyAndroidBlogs baiqiantao baiqiantao bqt20094 baiqiantao@sina.com

目录

装饰模式

简介

装饰模式以对客户端透明的方式【扩展】对象的功能,客户端并不会觉得对象在装饰前和装饰后有什么不同。

PS:对客户端透明的意思是,因为装饰类和原始类实现了相同的接口,所以你只需像使用原始类一样使用装饰类即可。

换句话说,你不需要学习新知识,就能完全知道装饰类有什么用

装饰模式是【继承】关系的一个替代方案。

只为增加功能而使用继承,当基类较多时会导致继承体系越来越臃肿,装饰模式可以在不创造更多子类的情况下,将对象的功能加以扩展。

装饰对象和真实对象有相同的接口,这样客户端对象就可以以和真实对象相同的方式和装饰对象交互。

装饰对象包含一个真实对象的引用,它接收所有来自客户端的请求,并把这些请求转发给真实的对象,并在转发这些请求之前或之后可以附加一些功能。

角色

  • 抽象构件角色(Component):给出一个抽象接口,以规范准备接收附加责任的对象。
  • 具体构件角色(Concrete Component):定义将要接收附加责任的类。
  • 装饰角色(Decorator):持有一个构件(Component)对象的引用,并定义一个与抽象构件接口一致的接口。
  • 具体装饰角色(Concrete Decorator):负责给构件对象“贴上”附加的责任。

优点 :

  • 解决【类膨胀】【类爆炸】【继承体系臃肿】的问题
  • 扩展性非常好
  • 比继承更灵活

案例

原始接口只具有work功能:

interface IWork {
public void work();
}

此接口有很多不同类型的实现类:

class People implements IWork {
@Override
public void work() {
System.out.println("人开始干活.....................");
}
} class Robot implements IWork {
@Override
public void work() {
System.out.println("机器人开始干活.....................");
}
} class AI implements IWork {
@Override
public void work() {
System.out.println("AI开始干活.....................");
}
}

如果使用继承来扩展原始类的功能:

class PeopleV2 extends People {
@Override
public void work() {
System.out.println("先检查零件是否正常");//增强的功能
super.work();
System.out.println("干完活收拾下房间\n");//增强的功能
}
} class RobotV2 extends Robot {
@Override
public void work() {
System.out.println("先检查零件是否正常");//增强的功能
super.work();
System.out.println("干完活收拾下房间\n");//增强的功能
}
} class AIV2 extends AI {
@Override
public void work() {
System.out.println("先检查零件是否正常");//增强的功能
super.work();
System.out.println("干完活收拾下房间\n");//增强的功能
}
}

可以看到,为了在需要的时候(注,不能直接更改原始类的work()方法,因为这些增强功能不是任何时候都需要的)增强原始类的功能,需要为每IWork接口的所有实现类都添加一个子类,在子类中实现对原始类功能的增强。当IWork接口的实现类很多时,会造成类体系迅速膨胀!

如果使用装饰模式来扩展原始类的功能,则只需定义一个装饰角色类Decorator:

class Decorator implements IWork {
private IWork work;
public Decorator(IWork work) {
this.work = work;
} @Override
public void work() {
System.out.println("先检查零件是否正常");//增强的功能
work.work();
System.out.println("干完活收拾下房间\n");//增强的功能
}
}

动态地给对象添加一些额外的职责。就增加功能来说,Decorator模式相比生成子类更为灵活。

客户端调用

public class Test {
public static void main(String args[]) {
IWork work = new People();
Decorator decorator = new Decorator(work);
decorator.work();
}
}

2016-10-31

Decorator Wrapper 装饰模式 MD的更多相关文章

  1. Decorator Wrapper 装饰模式 包装

    简介 装饰模式 装饰模式以对客户端[透明]的方式[扩展]对象的功能,客户端并不会觉得对象在装饰前和装饰后有什么不同,是继承关系的一个替代方案. 若只为增加功能而使用继承,当基类较多时会导致继承体系越来 ...

  2. python's decorator&wrapper

    [python's decorator&wrapper] decorator A function returning another function, usually applied as ...

  3. Decorator Pattern(装饰模式)

    装饰模式:动态的给一个对象添加一些额外的职责.当然我们也可以通过继承来实现类似的功能,但是随着子类的增多,各种子类的组合会造成子类的急剧膨胀. Requirement: 假设客户有一个要求,需要打一个 ...

  4. 深入浅出设计模式——装饰模式(Decorator Pattern)

    模式动机 一般有两种方式可以实现给一个类或对象增加行为: 继承机制,使用继承机制是给现有类添加功能的一种有效途径,通过继承一个现有类可以使得子类在拥有自身方法的同时还拥有父类的方法.但是这种方法是静 ...

  5. 第 13 章 装饰模式【Decorator Pattern】

    以下内容出自:<<24种设计模式介绍与6大设计原则>> Ladies and gentlemen,May I get your attention,Please?,Now I’ ...

  6. Decorator学习笔记

    初学者,自己的理解,请各位前辈不吝指正! Decorator,装饰模式,设计模式之一,谈谈我的理解,装饰这个词在我概念中就是给某个事物加上一些美丽的外表,把它变得更加完美.但是装饰是可以随时改变的,可 ...

  7. 《Java设计模式》之装饰模式

    装饰模式(Decorator) 1.    装饰模式(Decorator)的定义:又名包装(Wrapper)模式.装饰模式以对client透明的方式扩展对象的功能,是继承关系的一个替代方案. 2.   ...

  8. 12、Decorator 装饰器 模式 装饰起来美美哒 结构型设计模式

    1.Decorator模式 装饰模式又名包装(Wrapper)模式.装饰模式以对客户端透明的方式扩展对象的功能,是继承关系的一个替代方案. 装饰器模式(Decorator Pattern)允许向一个现 ...

  9. Python Decorator分析

    decorator本身是一个函数,这个函数的功能是接受被修饰的函数(decorated)作为参数,返回包装函数(wrapper)替换被修饰函数(decorated). @decorator func ...

随机推荐

  1. 哪种写法更好?<script></script> vs/or <script type=”text/javasript”></script>

    一直很奇怪 哪种写法更好<script type=“text/javascript”>…</script> or <script>…</script>? ...

  2. python functools.wraps

    我们在使用装饰器的时候,有些函数的功能会丢失,比如func.__name__,func.__doc__,func.__module__ 比如下面这个例子: In [16]: def logged(fu ...

  3. django项目添加新的app

  4. Java实现杨辉三角形

    Java实现杨辉三角形 一.源代码:YFTriangle.java package cn.com.zfc.day009; import java.util.Scanner; /** * 打印杨辉三角形 ...

  5. LCT维护子树信息

    有些题目,在要求支持link-cut之外,还会在线询问某个子树的信息.LCT可以通过维护虚边信息完成这个操作. 对于LCT上每个节点,维护两个两sz和si,后者维护该点所有虚儿子的信息,前者维护该点的 ...

  6. HTML5区块和大纲算法

    原文链接: Using HTML sections and outlines - Mozilla Developer Network 每集HTML5+CSS3网页布局教程-2大纲算法 HTML5标准带 ...

  7. hdu 4435 第37届ACM/ICPC天津现场赛E题

    转载请注明出处,谢谢http://blog.csdn.net/ACM_cxlove?viewmode=contents    by---cxlove 题目:给出N个城市,从1开始需要遍历所有点,选择一 ...

  8. 【BZOJ】4152: [AMPPZ2014]The Captain【SLF优化Spfa】

    4152: [AMPPZ2014]The Captain Time Limit: 20 Sec  Memory Limit: 256 MBSubmit: 2107  Solved: 820[Submi ...

  9. 【BZOJ-3110】K大数查询 整体二分 + 线段树

    3110: [Zjoi2013]K大数查询 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 6265  Solved: 2060[Submit][Sta ...

  10. 简单的php自定义错误日志

    平时经常看php的错误日志,很少有机会去自己动手写日志,看了王健的<最佳日志实践>觉得写一个清晰明了,结构分明的日志还是非常有必要的. 在写日志前,我们问问自己:为什么我们有时要记录自定义 ...