Java设计模式学习记录-装饰模式
前言
装饰模式也是一种结构型模式,主要是目的是相对于类与类之间的继承关系来说,使用装饰模式可以降低耦合度。JDK中有不少地方都使用到了装饰模式,例如Java的各种I/O流,javax.swing包中一些图形界面构件功能的增强等地方都运用了装饰模式。
装饰模式
定义
装饰模式的定义是:在不改变原类文件以及不使用继承的情况下,动态的扩展一个对象的功能。装饰模式是通过创建一个包装对象来实现的,也就是用装饰来包裹真实的对象。
举例
还是老规矩,举例说明,在给亲朋好友过生日时会买生日蛋糕,然后生日蛋糕又有各种各样的辅料来进行装饰,例如:奶油,水果,芝士,巧克力等等。如果没有这些辅料来进行装饰,就是普通的鸡蛋糕。所以我们可以把做蛋糕的这个过程来用代码实现出来:
首先定义蛋糕接口
/**
* 蛋糕
*/
public interface CakeGoods { /**
* 展示蛋糕
*/
void showCake(); /**
* 展示价格
*/
BigDecimal showPrice();
}
具体的鸡蛋糕
/**
* 鸡蛋糕
*/
@Data
public class SpongeCake implements CakeGoods{ //蛋糕名称
private String name;
//蛋糕价格
private BigDecimal price; public SpongeCake(String name,BigDecimal price){ this.name = name;
this.price = price;
} /**
* 展示蛋糕
*/
@Override
public void showCake() { System.out.println("的"+this.name+"蛋糕"); } /**
* 展示价格
*/
@Override
public BigDecimal showPrice() { return this.price;
}
}
辅料的抽象类
/**
* 辅料的抽象类
*/
public abstract class Decorator implements CakeGoods{ private CakeGoods cakeGoods; public void setCakeGoods(CakeGoods cakeGoods){
this.cakeGoods = cakeGoods;
} /**
* 展示蛋糕
*/
@Override
public void showCake() {
cakeGoods.showCake();
} /**
* 展示价格
*/
@Override
public BigDecimal showPrice() {
return cakeGoods.showPrice();
}
}
水果
/**
* 水果
*/
public class Fruits extends Decorator {
/**
* 展示蛋糕
*/
@Override
public void showCake() {
System.out.print("加水果");
super.showCake();
} /**
* 展示价格
*/
@Override
public BigDecimal showPrice() {
return new BigDecimal(20.00).add(super.showPrice());
}
}
奶油
/**
* 奶油
*/
public class Cream extends Decorator {
/**
* 展示蛋糕
*/
@Override
public void showCake() {
System.out.print("加奶油");
super.showCake();
} /**
* 展示价格
*/
@Override
public BigDecimal showPrice() {
return new BigDecimal(15.00).add(super.showPrice());
}
}
测试
public class Tests { public static void main(String[] args) { //制造鸡蛋糕
CakeGoods cakeGoods = new SpongeCake("生日祝福",new BigDecimal(50)); Decorator cream = new Cream();
Decorator fruits = new Fruits(); //加奶油
cream.setCakeGoods(fruits);
//加水果
fruits.setCakeGoods(cakeGoods); //展示
cream.showCake();
System.out.println("的价格是:"+cream.showPrice()+"元");
} }
运行结果:
加奶油加水果的生日祝福蛋糕
的价格是:85元
上面的这个蛋糕制造的例子使用的就是装饰模式,在为鸡蛋糕SpongeCake进行扩展的时候并没有影响它原来的类的结构,也没有使用继承的关系,最终却达到了装饰的目的。下面我们来分析一下装饰模式具体是由那几部分组成。
装饰模式的结构
装饰模式的结构图如上所示,主要由以下几个角色组成。
抽象构件角色:(上图的CakeGoods)定义一个抽象接口,以规范准备接受装饰的对象,想当于Java中的IO流里的InputStram/OutputStream和Reader/Writer。
具体的构件角色:(上图的SpongeCake)定义一个将要接受装饰的类,相当于I/O流里面的FileOutputStream和FileInputStream。
装饰角色:(上图的Decorator)定义一个持有抽象构件角色的引用,并定义一个与抽象构件一直的接口。相当于I/O里面的FilterOutputStream和FilterInputStream。
具体的装饰角色:(上图的Fruits和Cream)负责各构件角色对象加上相应的装饰品,相当于I/O流里的BufferedOutputStream、BufferedInputStream。
在使用的时候,必须扩展CakeGoods的功能,但是我们是通过Fruits和Cream来对CakeGoods进行扩展的,所以相对于CakeGoods来说无需知道Decorator的存在,就能扩展功能了。
总结
装饰模式的优点
- 对于扩展一个对象的功能,装饰模式比继承更加灵活性,不会导致类的个数急剧增加。例如上面的例子如果想增加一个枣糕或是一个巧克力辅料,无需修改现有代码,只需将枣糕作为CakeGoods的一个子类,以及Decorator的一个子类即可。
- 可以动态的扩展一个对象的功能。
- 可以对一个对象进行多次装饰,通过使用不同的具体装饰类以及这些装饰类的排列组合,可以创造出很多不同行为的组合,得到功能更为强大的对象。
- 具体构件类与具体装饰类可以独立变化,用户可以根据需要增加新的具体构件类和具体装饰类,原有类库代码无须改变,符合“开闭原则”。
装饰模式的缺点
- 使用装饰模式进行系统设计时将产生很多小对象,这些对象的区别在于它们之间相互连接的方式有所不同,而不是它们的类或者属性值有所不同,大量小对象的产生势必会占用更多的系统资源,在一定程序上影响程序的性能。
- 装饰模式提供了一种比继承更加灵活机动的解决方案,但同时也意味着比继承更加易于出错,排错也很困难,对于多次装饰的对象,调试时寻找错误可能需要逐级排查,较为繁琐。
适用场景
在不影响其他对象的情况下,以动态、透明的方式给单个对象添加职责。
当不能采用继承的方式对系统进行扩展或者采用继承不利于系统扩展和维护时可以使用装饰模式。不能采用继承的情况主要有两类:第一类是系统中存在大量独立的扩展,为支持每一种扩展或者扩展之间的组合将产生大量的子类,使得子类数目呈爆炸性增长;第二类是因为类已定义为不能被继承(如Java语言中的final类)。
加油,给自己打气,克服惰性!!!
想了解更多的设计模式请查看Java设计模式学习记录-GoF设计模式概述。
Java设计模式学习记录-装饰模式的更多相关文章
- Java设计模式学习记录-迭代器模式
前言 这次要介绍的是迭代器模式,也是一种行为模式.我现在觉得写博客有点应付了,前阵子一天一篇,感觉这样其实有点没理解透彻就写下来了,而且写完后自己也没有多看几遍,上次在面试的时候被问到java中的I/ ...
- Java设计模式学习记录-模板方法模式
前言 模板方法模式,定义一个操作中算法的骨架,而将一些步骤延迟到子类中.使得子类可以不改变一个算法的结构即可重新定义该算法的某些特定步骤. 模板方法模式 概念介绍 模板方法模式,其实是很好理解的,具体 ...
- Java设计模式学习记录-状态模式
前言 状态模式是一种行为模式,用于解决系统中复杂的对象状态转换以及各个状态下的封装等问题.状态模式是将一个对象的状态从该对象中分离出来,封装到专门的状态类中,使得对象的状态可以灵活多变.这样在客户端使 ...
- Java设计模式学习记录-观察者模式
前言 观察者模式也是对象行为模式的一种,又叫做发表-订阅(Publish/Subscribe)模式.模型-视图(Model/View)模式. 咱们目前用的最多的就是各种MQ(Message Queue ...
- Java设计模式学习记录-备忘录模式
前言 这次要介绍的是备忘录模式,也是行为模式的一种 .现在人们的智能手机上都会有备忘录这样一个功能,大家也都会用,就是为了记住某件事情,防止以后自己忘记了.那么备忘录模式又是什么样子的呢?是不是和手机 ...
- Java设计模式学习记录-解释器模式
前言 这次介绍另一个行为模式,解释器模式,都说解释器模式用的少,其实只是我们在日常的开发中用的少,但是一些开源框架中还是能见到它的影子,例如:spring的spEL表达式在解析时就用到了解释器模式,以 ...
- Java设计模式学习记录-命令模式
前言 这次要介绍的是命令模式,这也是一种行为型模式.最近反正没有面试机会我就写博客呗,该投的简历都投了.然后就继续看书,其实看书也会给自己带来成就感,原来以前不明白的东西,书上已经给彻底的介绍清楚了, ...
- Java设计模式学习记录-享元模式
前言 享元模式也是一种结构型模式,这篇是介绍结构型模式的最后一篇了(因为代理模式很早之前就已经写过了).享元模式采用一个共享来避免大量拥有相同内容对象的开销.这种开销最常见.最直观的就是内存损耗. 享 ...
- Java设计模式学习记录-外观模式
前言 这次要介绍的是外观模式(也称为门面模式),外观模式也属于结构型模式,其实外观模式还是非常好理解的,简单的来讲就是将多个复杂的业务封装成一个方法,在调用此方法时可以不必关系具体执行了哪些业务,而只 ...
随机推荐
- C++静态库与动态库(比较透彻)
这次分享的宗旨是——让大家学会创建与使用静态库.动态库,知道静态库与动态库的区别,知道使用的时候如何选择.这里不深入介绍静态库.动态库的底层格式,内存布局等,有兴趣的同学,推荐一本书<程序员的自 ...
- squid常用操作
如何查看squid的缓存命中率 使用命令: squidclient -h host -p port mgr:info比如: /usr/local/squid/bin/squidclient -h 12 ...
- tensorflow学习之(五)构造简单神经网络 并展示拟合过程
# def 添加层 如何构造神经网络 并展示拟合过程 import tensorflow as tf import numpy as np import matplotlib.pyplot as pl ...
- vue全局路由守卫beforeEach
在main.js里使用方法 router.beforeEach((to,from,next)=>{}) to,是将要跳转的路由, from,是离开的路由 next是个方法,判断to.path 或 ...
- 《MarkMark学习笔记学习笔记》html学习笔记
iframe里有一个srcdoc属性,很有用! window.location.href=document.referrer//可以实现返回上一级页面并刷新 HTML5权威指南©®,比较老的书了,有些 ...
- 第一节 —— vue2.0 环境安装,工程化开发
vue的开发有两种,一种是直接的在script标签里引入vue.js文件即可,这样子引入的话个人感觉做小型的多页面会比较舒坦,一旦做大型一点的项目,还是离不开webpack. 所以另一种方法也就是基于 ...
- UniDBGrid增加显示记录数的label及隐藏refresh按钮
1. 在UniDBgrid的extEvent属性中写入以下代码: function OnAfterCreate(sender){ var toolbar=sender.getDockedItems() ...
- LeNet,AlexNet,GoogleLeNet,VggNet等网络对比
CNN的发展史 上一篇回顾讲的是2006年Hinton他们的Science Paper,当时提到,2006年虽然Deep Learning的概念被提出来了,但是学术界的大家还是表示不服.当时有流传的段 ...
- 项目Alpha冲刺(团队5/10)
项目Alpha冲刺(团队5/10) 团队名称: 云打印 作业要求: 项目Alpha冲刺(团队) 作业目标: 完成项目Alpha版本 团队队员 队员学号 队员姓名 个人博客地址 备注 221600412 ...
- 宽字符————_T、_TEXT、L、TEXT之间的区别
_T._TEXT.L.TEXT之间的区别 在分析前先对三者做一个简单的分类 _T._TEXT.TEXT三者都是根据编译器的环境进行ANSI/UNICODE变换的,_T和_TEXT是根据_UNICODE ...