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. Java 中的三大特性

    我们都知道 Java 中有三大特性,那就是继承 ,封装和多态 .那我今天我就来说说这几个特性 . 老样子 ,先问问自己为什么会存在这些特性 .首先说封装 ,封装就是使用权限修饰符来实现对属性的隐藏 , ...

  2. [代码审计]DM企业建站系统v201710 sql注入漏洞分析 | 新版v201712依旧存在sql注入

    0x00 前言 本来呢,这套CMS都不想审的了.下载下来打开一看,各种debug注释,排版烂的不行. 贴几个页面看看 感觉像是新手练手的,没有审下去的欲望了. 但想了想,我tm就是新手啊,然后就继续看 ...

  3. Xshell拖拽上传文件插件

    lrzsz是一款在linux里可代替ftp上传和下载的程序.在linux中支持直接拖拽上传的插件:同时也支持rz和sz进行命令上传和下载. 插件安装 yum -y install lrzsz 上传(r ...

  4. [leetcode DP]72. Edit Distance

    计算最少用多少不把word1变为word2, 思路:建立一个dp表,行为word1的长度,宽为word2的长度 1.边界条件,dp[i][0] = i,dp[0][j]=j 2.最优子问题,考虑已经知 ...

  5. springboot项目打war包部署到服务器(eclipse & gradle环境)

    1.右键项目run as  -> run configurations,打开Run configurations弹出框 2.右键Gradle Project -> New Configur ...

  6. Alpha 冲刺报告8

    组长:吴晓晖 今天完成了哪些任务: maven和idea用的不熟啊,jar包或者war包导出来一直有问题:生气了把ide扔到服务器里去运行springboot了,卡哭了,终于可以运行了,然后debug ...

  7. 【10.4校内测试】【轮廓线DP】【中国剩余定理】【Trie树+博弈】

    考场上几乎是一看就看出来轮廓线叻...可是调了两个小时打死也过不了手出样例!std发下来一对,特判对的啊,转移对的啊,$dp$数组竟然没有取max!!! 某位考生当场死亡. 结果下午又请了诸位dala ...

  8. bzoj 2011

    决策单调性,对于一个1D/1D(状态是一维,转移也是一维)的DP,如果DP的决策具有单调性,那么就可以做到O(nlogn)的复杂度完成DP. 感谢<1D/1D  动态规划优化初步>的作者. ...

  9. Zookeeper的基本操作

    写在前面的话:读书破万卷,编码如有神 -------------------------------------------------------------------- 参考内容: <私塾 ...

  10. 预防Redis缓存穿透、缓存雪崩解决方案

    最近面试中遇到redis缓存穿透.缓存雪崩等问题,特意了解下. redis缓存穿透: 缓存穿透是指用户查询数据,在数据库没有,自然在缓存中也不会有.这样就导致用户查询的时候,在缓存中找不到,每次都要去 ...