Java IO设计模式(装饰模式与适配器模式)
01. 装饰模式
1. 定义
Decorator装饰器,就是动态地给一个对象添加一些额外的职责,动态扩展,和下面继承(静态扩展)的比较。因此,装饰器模式具有如下的特征:
- 它必须持有一个被装饰的对象(作为成员变量)。
- 它必须拥有与被装饰对象相同的接口(多态调用、扩展需要)。
- 它可以给被装饰对象添加额外的功能。
总结:保持接口,动态增强性能。
装饰器通过包装一个装饰对象来扩展其功能,而又不改变其接口,这实际上是基于对象的适配器模式的一种变种。与对象的适配器模式异同:
- 相同点:都拥有一个目标对象。
- 不同点:适配器模式需要实现旧接口,而装饰器模式必须实现相同接口。
适配器模式是在适配器中,重写旧接口的方法来调用新接口方法,来实现旧接口不改变,同时使用新接口的目的。新接口适配旧接口。
而装饰模式,是装饰器和旧接口实现相同的接口,在调用新接口的方法中,会调用旧接口的方法,并对其进行扩展。
2. 由来(为什么不是继承)
功能的拓展,通常可以使用继承的方式解决。但这样实现的话,每一种组合都需要一个类,大量重复性内容,类数目“爆炸”;另外,这些拓展的功能必须要是可以预见,编译时就确定了,静态的扩展。
一个例子大概说:Beverage是一个抽象类,它被所有在一个咖啡店里卖的饮料继承。Beverage有个抽象方法cost,所有的子类都要实现这个抽象方法,计算它们的价格。现在有四个最基本的咖啡:HouseBlend,DarkRoast,Decaf,Espresso他们都继承自Beverage,现在的需求是说在四个最基本的咖啡里,每个都可以随便地添加调味品,像steamed milk,soy,还有mocha最后是加上whipped milk。如果是说按继承来实现这种几个调味品跟原来咖啡的组合的话,我们会很自然地设计来下面的类图来:
如果是按装饰模式的设计思路我们可以得出下面的设计类图:
装饰模式是怎么达到不仅类的数目大减少了,性能的重复也可以减至到最少。
3. 典型结构图
一句话解释:装饰者和被装饰者需要继承同一个接口或者是抽象类,被装饰者作为装饰者的一个变量。程序中原来调用被装饰者某方法func1的地方改成调用装饰者相同的那个方法func1,并且装饰者的该方法func1上添加了一些额外的功能,在方法func1中再调用被装饰着的方法func1。
这就是动态的扩展。
02. 适配器模式
目的:将一个类的接口转换成客户期望的另一个接口,让原本不兼容的接口可以合作无间。
1. 特点
- 适配器对象实现原有接口
- 适配器对象组合一个实现新接口的对象(这个对象也可以不实现一个接口,只是一个单纯的对象)
- 对适配器原有接口方法的调用被委托给新接口的实例的特定方法(重写旧接口方法来调用新接口功能。)
2.例子
我国国标充电器三孔,德国得标充电器两孔。现去德国旅行。如何将我们的三孔充电器插入两孔。这就需要适配器。
类图:
DBSocketInterface:德标接口
DBSocket:德国插座(实现DBSocketInterface,提供两孔充电方法)
Hotel : 拥有得标接口。
GBSocketInterface :国标接口
GBSocket : 中国插座(实现GBSocketInterface,提供三孔充电方法)
适配器实现:
public class SocketAdapter
implements DBSocketInterface{ //实现旧接口
//组合新接口
private GBSocketInterface gbSocket;
/**
* 在创建适配器对象时,必须传入一个新街口的实现类
*/
public SocketAdapter(GBSocketInterface gbSocket) {
this.gbSocket = gbSocket;
}
/**
* 将对就接口的调用适配到新接口
*/
@Override
public void powerWithTwoRound() {
gbSocket.powerWithThreeFlat();
}
}
在适配器中,继承了旧接口,组合了新接口。在重写旧接口方法的时候,调用了新接口的功能。
hotel.setSocket(socketAdapter);
hotel.charge();
在hotel的charge()方法中,调用的是DBSocketInterface 实现子类的两孔充电方法。而这个实现子类,就是适配器类,重写的调用三孔充电的方法。
这也是适配器模式魅力:
不改变原有接口(德标),却还能使用新接口的功能(国标)。
适配器详情和实际使用例子参考《http://blog.csdn.net/zhangjg_blog/article/details/18735243》
03. Java IO 设计模式
1. I/O库对称性
- 输入-输出对称:比如InputStream和OutputStream对Byte字节流的输入和输;而Reader和Writer各自占据Char字符流的输入和输出。
- byte-char对称:InputStream和Reader的子类分别负责byte和字符流的输入;OutputStream和Writer的子类分别负责byte和字符流的输出。
2.两个设计模式
- 装饰模式:在由InputStream、OutputStream、Reader和Writer代表的等级结构内部,有一些流处理器可以对另一些流处理器起到装饰作用,形成新的、具有改善了的功能的流处理器。
- 适配器模式:在由InputStream、OutputStream、Reader和Writer代表的等级结构内部,有一些流处理器是对其他类型的流处理器的适配。这就是适配器的应用。
3.装饰模式的应用
由于java I/O库需要很多性能的各种组合,如果这些性能都是用继承来实现,那么每一种组合都需要一个类,大量重复类。
首先,需要理解java I/O库是由一些原始流处理器和围绕它的装饰流处理器(装饰器,动态扩展原始类性能)所组成的。
这里以InputStream为例,并附上结构图。
这些流类分成两种,即原始流类(Original Stream)和链接流处理器(Wrapper Stream)。
原始流处理器
原始流处理器接收一个Byte数组对象,String对象,FileDiscriptor对象或者不同类型的流源对象,原始流处理器包括以下四种:
- ByteArrayInputStream:接收一个Byte数组作为流的源。
- FileInputStream:建立一个与文件有关的输入流。接收一个File对象作为流的源。
- PipedInputStream:可以与PipedOutputStream配合使用,用于读入一个数据管道的数据,接收一个PipedOutputStream作为源。
- StringBufferInputStream:将一个字符串缓冲区转换为一个输入流。(废弃)
链接流处理器
所谓链接流处理器,就是可以接收另一个流对象作为源,并对之进行功能扩展的类。InputStream类型的链接处理接收另一个InputStream对象作为流源。
以FilterInputStream过滤输入流的子类为例。它将另一个输入流作为流源。这个类的子类包括以下几种:
- BufferedInputStream:用来从硬盘将数据读入到一个内存缓冲区中,并从缓冲区提供数据。
- DataInputStream:提供基于多字节的读取方法,可以读取原始类型的数据。
- LineNumberInputStream:提供带有行计数功能的过滤输入流。
- PushbackInputStream:提供特殊的功能,可以将已经读取的字节“推回”到输入流中。
原始流,就是装模式中具体构件角色(被装饰者),链接流,就是装饰模式中的装饰角色。
抽象结构图
一句话总结:链接流处理器,接收原始流处理器并将其作为成员变量引用,都继承了相同的抽象类InputStream。他们在内部工作方法中做了相应改变,在链接流的相同方法中扩展原始流中的方法,这种变化就是装饰模式的目的。
4.适配器模式的应用
StringBufferInputStream是一个适配器类,其继承了InputStream类型,同时持有一个对String类型的引用。这是将处理String对象的新接口适配成InputStream的旧接口的适配器模式。
其余OutputStream、Reader和Writer的装饰模式和适配器模式
参考《http://www.cnblogs.com/wxgblogs/p/5649933.html》
和《http://www.cnblogs.com/heartstage/p/3391070.html》
Java IO设计模式(装饰模式与适配器模式)的更多相关文章
- Java IO设计模式
JAVA IO 设计模式彻底分析 2011-01-06 14:20:09| 分类: java|字号 订阅 http://blog.csdn.net/tianyue168/archive/2010/0 ...
- Java IO设计模式彻底分析 (转载)
一.引子(概括地介绍Java的IO) 无论是哪种编程语言,输入跟输出都是重要的一部分,Java也不例外,而且Java将输入/输出的功能和使用范畴做了很大的扩充.它采用了流的 机制来实现输入/输出,所谓 ...
- Java程序设计模式系列之适配器模式
理解适配器设计模式需要从生活中的场景进行联系,在生活当中有那些东西能够称为适配器呢?从字面上理解,"适配"的意思就是让一个东西和另一个东西配对,能够让他们一起工作,比如大家用的笔记 ...
- java常用设计模式六:适配器模式
一.定义 适配器模式把一个类的接口变换成客户端所期待的另一种接口,从而使原本因接口不匹配而无法在一起工作的两个类能够在一起工作.比如以下的场景: 用手机充电为例,有一个手机的插孔是TypeC口,现在只 ...
- Java描述设计模式(07):适配器模式
本文源码:GitHub·点这里 || GitEE·点这里 一.适配器模式简介 1.基础概念 适配器模式把一个类的接口变换成客户端所期待的另一种接口,从而使原本因接口不匹配而无法在一起工作的两个类能够在 ...
- JAVA IO 序列化与设计模式
➠更多技术干货请戳:听云博客 序列化 什么是序列化 序列化:保存对象的状态 反序列化:读取保存对象的状态 序列化和序列化是Java提供的一种保存恢复对象状态的机制 序列化有什么用 将数据保存到文件或数 ...
- java.io包下适配和装饰模式的使用
如java.io.LineNumberInputStream(deprecated),是装饰模式(decorate)的实现: 如java.io.OutputStreamWriter,是适配器模式(ad ...
- java常用设计模式总览
一.java的设计模式大体上分为三大类: 创建型模式(5种):工厂方法模式,抽象工厂模式,单例模式,建造者模式,原型模式. 结构型模式(7种):适配器模式,装饰器模式,代理模式,外观模式,桥接模式,组 ...
- Java IO流学习
Java IO流学习 Java流操作有关的类或接口: Java流类图结构: 流的概念和作用 流是一组有顺序的,有起点和终点的字节集合,是对数据传输的总称或抽象.即数据在两设备间的传输称为流,流的本质是 ...
随机推荐
- 学习笔记TF016:CNN实现、数据集、TFRecord、加载图像、模型、训练、调试
AlexNet(Alex Krizhevsky,ILSVRC2012冠军)适合做图像分类.层自左向右.自上向下读取,关联层分为一组,高度.宽度减小,深度增加.深度增加减少网络计算量. 训练模型数据集 ...
- (数字IC)低功耗设计入门(七)——门级电路低功耗设计优化(续)
前面讲解了门级功耗的优化方法,包括静动态和总体的功耗.现在来记录一下门级层次(有点书也说是在系统级)常用的一种低功耗方法--电源门控. ①电源门控概述与原理 电源门控是指芯片中某个区域的供电电源被关掉 ...
- python之 centos6.7下 python 3.5.2、Django-1.9 安装
在linux6.5中已经自带了python 2 .python 2.6 ,并且yum程序使用的就是自带的python,所以系统自带的python不要随意卸载否则可能导致yum用不了. 测试环境:cen ...
- 点评阿里JAVA手册之异常日志(异常处理 日志规约 )
下载原版阿里JAVA开发手册 [阿里巴巴Java开发手册v1.2.0] 本文主要是对照阿里开发手册,注释自己在工作中运用情况. 本文内容:异常处理 日志规约 本文难度系数为一星(★) 本文为第三篇 ...
- leetcode算法题(JavaScript实现)
题外话 刷了一段时间的codewars的JavaScript题目之后,它给我最大的感受就是,会帮助你迅速的提升你希望练习的语言的API的熟悉程度,Array对象.String对象等原生方法,构造函数. ...
- mysql之 MySQL 主从基于 GTID 复制原理概述
一. 什么是GTID ( Global transaction identifiers ):MySQL-5.6.2开始支持,MySQL-5.6.10后完善,GTID 分成两部分,一部分是服务的UUid ...
- Hybrid App开发之jQuery基础
前言: 前面学习了JavaScript/Html/Css的基础知识,今天学习一下常用js框架jQuery的使用进行快速的开发. JQuery的基本功能: 方位和操作DOM元素 控制页面样式 对页面事件 ...
- PHP:phpMyAdmin如何解决本地导入文件(数据库)为2M的限制
经验地址:http://jingyan.baidu.com/article/e75057f2a2288eebc91a89b7.html 当我们从别人那里导出数据库在本地导入时,因为数据库文件大于2M而 ...
- WPF MVVM 架构 Step By Step(6)(把actions从view model解耦)
到现在为止,我们创建了一个简单的MVVM的例子,包含了实现了的属性和命令.我们现在有这样一个包含了例如textbox类似的输入元素的视图,textbox用绑定来和view model联系,像点击but ...
- IPhone开发“此证书是由未知颁发机构签名”解决办法
有一种情况是你删除了钥匙串中的系统文件,只要重新下载,并双击(会自动添加到钥匙串中)就ok了. 从浏览器中直接敲入下载地址:http://developer.apple.com/certificati ...