★ 场景和问题

在不对原有对象类进行修改的基础上,如何给一个或多个已有的类对象提供增强额外的功能?

★ 引例

写一个MyBufferedReader类,使它能够对字符流(如FileReader、InputStreamReader和PipedReader等)进行功能增强:

(1) 提供带缓冲的myRead()方法,对原有的read()方法进行增速;

(2)提供一个能够每次读取一行字符的myReadLine()方法。

实现对FileReader的功能加强:

package cn.hncu.pattern.decorator.v1;

import java.io.FileReader;
import java.io.IOException; public class MyBufferedReader {
private FileReader fr;//封装
private char buf[] = new char[1024];
private int count = 0;// 记录当前缓冲区中的字符个数
private int pos = 0;// 游标,数组下标,当前读取的是数组中的哪个元素 public MyBufferedReader(FileReader fr){
this.fr=fr;
} public int MyRead() throws IOException{
// 当缓冲区为空时,用fr对象到文件中去读取一组数据到缓冲区中
if(count==0){
count = fr.read(buf);
if(count==-1){
return -1;
}
pos=0;;
} // 从缓冲区中取一个字符出去
char ch = buf[pos];
pos++;
count--;
//返回那个字符
return ch;
} public String MyReadLine() throws IOException{
StringBuffer strBuffer = new StringBuffer();
int ch=0;
while(((ch=MyRead())!=-1)){
if(ch=='\r'){
continue;
}
if(ch=='\n'){
return strBuffer.toString();
} strBuffer.append(ch);
} if(strBuffer.length()!=0){
return strBuffer.toString();
}
return null;
} public void MyClose() throws IOException{
fr.close();
} }

测试类:

package cn.hncu.pattern.decorator;

import java.io.BufferedReader;
import java.io.FileNotFoundException;
import java.io.FileReader;
import java.io.IOException;
import java.io.InputStream; import cn.hncu.pattern.decorator.v1.MyBufferedReader; public class TestMyBufferReader { public static void main(String[] args) throws IOException {
//testMyBufferedReader();//测试自己写的缓存流 testBufferedReader();//看看系统的缓存流读取的是不是和自己读取的相同
} //测试JavaAPI中的BufferedReader类
private static void testBufferedReader() throws IOException {
FileReader in = new FileReader("chx.txt");
BufferedReader bf = new BufferedReader(in);
int ch=0;
while((ch=bf.read())!=-1){
System.out.print((char)ch);
}
bf.close(); } //测试自己写的MyBufferedReader类
private static void testMyBufferedReader() throws IOException {
FileReader fr = new FileReader("chx.txt");
MyBufferedReader mbf = new MyBufferedReader(fr);
int ch=0;
while((ch=mbf.MyRead())!=-1){
System.out.print((char)ch);
} mbf.MyClose();
} }

文件chx.txt:

写一个MyBufferedReader类,使它能够对字符流
(如FileReader、InputStreamReader和PipedReader等)进行功能增强:
(1) 提供带缓冲的myRead()方法,对原有的read()方法进行增速;
(2)提供一个能够每次读取一行字符的myReadLine()方法。
chx!!!
OK.

测试类中testMyBufferedReader()方法的输出结果:

写一个MyBufferedReader类,使它能够对字符流
(如FileReader、InputStreamReader和PipedReader等)进行功能增强:
(1) 提供带缓冲的myRead()方法,对原有的read()方法进行增速;
(2)提供一个能够每次读取一行字符的myReadLine()方法。
chx!!!
OK.

测试类中testBufferReader()方法的输出结果:

写一个MyBufferedReader类,使它能够对字符流
(如FileReader、InputStreamReader和PipedReader等)进行功能增强:
(1) 提供带缓冲的myRead()方法,对原有的read()方法进行增速;
(2)提供一个能够每次读取一行字符的myReadLine()方法。
chx!!!
OK.

可以看到,输出的结果是一样的!

下面实现对多个类的缓存增强支持:

package cn.hncu.pattern.decorator.v3;

import java.io.IOException;
import java.io.Reader; public class MyReader extends Reader{
//如果不写继承,就没有融入体系中去,所以要继承Reader类
private Reader r;
private char[] buf = new char[1024];
private int count = 0;// 记录当前缓冲区中的字符个数
private int pos = 0;// 游标,数组下标,当前读取的是数组中的哪个元素 public MyReader(Reader r){
this.r=r;
} public int MyReader() throws IOException{
// 当缓冲区为空时,用r对象到文件中去读取一组数据到缓冲区中
if(count<=0){
count=0;
}
if(count==0){
count=r.read(buf);
if(count==-1){
return -1;
}
pos=0;
} // 从缓冲区中取一个字符出去
int ch = buf[pos];
pos++;
count--; return ch;
} public String MyReaderLine() throws IOException{
StringBuffer strBuffer = new StringBuffer();
int ch=0;
while((ch=MyReader())!=-1){
if(ch=='\r'){//回车
continue;
}
if(ch=='\n'){//换行
return strBuffer.toString();
}
char s = (char)ch;
strBuffer.append(s);
} if(strBuffer.length()!=0){
return strBuffer.toString();
}
return null;
} public void MyClose() throws IOException {
r.close();
} @Override
public int read(char[] cbuf, int off, int len) throws IOException {
return read(cbuf, off, len);
} @Override
public void close() throws IOException {
r.close();
} }

测试类:

package cn.hncu.pattern.decorator;

import java.io.FileReader;
import java.io.IOException;
import java.io.InputStreamReader; import cn.hncu.pattern.decorator.v3.MyReader; public class TestMyReader { public static void main(String[] args) throws IOException {
testMyReaderLine();//测试自己写的缓存流
System.out.println("-------------");
testMyReader();
} //测试自己写的MyReader类的MyReader()
private static void testMyReader() throws IOException {
InputStreamReader in = new FileReader("chx.txt");
MyReader mr = new MyReader(in);
int ch=0;
while((ch=mr.MyReader())!=-1){
System.out.print((char)ch);
}
mr.close(); } //测试自己写的MyReader类的MyReaderLine()
private static void testMyReaderLine() throws IOException {
FileReader fr = new FileReader("chx.txt");
MyReader mbf = new MyReader(fr);
String ch=null;
while((ch=mbf.MyReaderLine())!=null){
System.out.print(ch);
System.out.println();
} mbf.MyClose();
} }

文件和上面的文件是一样的,下面看输出结果:

写一个MyBufferedReader类,使它能够对字符流
(如FileReader、InputStreamReader和PipedReader等)进行功能增强:
(1) 提供带缓冲的myRead()方法,对原有的read()方法进行增速;
(2)提供一个能够每次读取一行字符的myReadLine()方法。
chx!!!
OK.
-------------
写一个MyBufferedReader类,使它能够对字符流
(如FileReader、InputStreamReader和PipedReader等)进行功能增强:
(1) 提供带缓冲的myRead()方法,对原有的read()方法进行增速;
(2)提供一个能够每次读取一行字符的myReadLine()方法。
chx!!!
OK.

装饰比继承更灵活,避免了继承造成的体系臃肿问题。

Java---设计模块(装饰模式)的更多相关文章

  1. Java 9 模块解耦的设计策略

    1. 概述 Java 平台模块系统 (Java Platform Module System,JPMS)提供了更强的封装.更可靠且更好的关注点分离. 但所有的这些方便的功能都需要付出代价.由于模块化的 ...

  2. Java 208 道面试题:Java 基础模块答案

    目前市面上的面试题存在两大问题:第一,题目太旧好久没有更新了,还都停留在 2010 年之前的状态:第二,近几年 JDK 更新和发布都很快,Java 的用法也变了不少,加上 Java 技术栈也加入了很多 ...

  3. Swing 是一个为Java设计的GUI工具包

    Swing 是一个为Java设计的GUI工具包. Swing是JAVA基础类的一部分. Swing包括了图形用户界面(GUI)器件如:文本框,按钮,分隔窗格和表. Swing提供许多比AWT更好的屏幕 ...

  4. 使用Java设计验证码生成程序

    我们来设计一个简单的验证码生成程序:验证码一个由4位的数字.字母随机组合而成图像,为了避免被光学字元识别(OCR,Optical Character Recognition)之类的程序识别出图片中的数 ...

  5. kettle工具的设计模块

    大家都知道,每个ETL工具都用不同的名字来区分不同的组成部分.kettle也不例外. 比如,在 Kettle的四大不同环境工具 本博客,是立足于kettle工具的设计模块的概念介绍. 1.转换 转换( ...

  6. 哈希表hashTable的Java设计

    1:哈希表的概念 2:设计原理 3:哈希表的Java设计

  7. Java 登录模块设计

    登录流程 前端登录传输用户名和md5加密后的密码 后端对密码在进行md5加密,或者使用md5加密的密码 + id 进行盐加密,增加密码被破解的难度. 登录成功后,这里分成单体,或者分布式的情况 单体 ...

  8. Java-单机版的书店管理系统(练习设计模块和思想_系列汇总)

    介绍: 本软件系列到此,我已经全部写完了. 项目练习目标 : 1.Java应用程序基本分析 2.培养面向对象编程的基本思想 3.Java基本设计模式综合应用 4.掌握分层和接口的基本设计 5.构建合理 ...

  9. MapDB:专为Java设计的高性能的数据库

    MapDB是一个快速.易用的嵌入式Java数据库引擎,它提供了基于磁盘或者堆外(off-heap允许Java直接操作内存空间, 类似于C的malloc和free)存储的并发的Maps.Sets.Que ...

  10. java设计原则:16种原则

    一   类的设计原则   1 依赖倒置原则-Dependency Inversion Principle (DIP) 2 里氏替换原则-Liskov Substitution Principle (L ...

随机推荐

  1. Android Studio常用插件

    ButterKnife 这个开源库可以让我们从大量的findViewById()和setonclicktListener()解放出来,其对性能的影响微乎其微(查看过Butter Knife的源码,其自 ...

  2. 日积月累:ProguardGui进行jar包代码混淆

    前面文章<Proguard进行源代码混淆>讲解过怎么使用Proguard工具对Android的源代码进行混淆的方法(感兴趣的朋友可以访问:http://blog.csdn.net/p106 ...

  3. android实现界面左右滑动(GridView动态设置item,支持每个item按某个属性排序来显示在不同的界面)

    效果图 :                         分别是第一页.第二页.第三页,随手截的图,不整齐,勿见怪.开始走了弯路,废了不少时间. 思路如下: 1.用ViewPager实现左右分页滑动 ...

  4. 【.NET特供-第三季】ASP.NET MVC系列:MVC与三层图形对照(颠覆性理论)

           在[.NET特供-第三季]系列博客中的第一篇<ASP.NET MVC系列:MVC与三层图形对照>发表之后,引起了领导的注意.同一时候,开发小组内部在交流MVC和三层之间关系的 ...

  5. Unity3D Asset stored 已下载的位置

    Unity3D Asset stored下载资源在本地的什么目录里呢?C:\Users\accountName\AppData\Roaming\Unity\Asset Store

  6. exploit writing tutorial 阅读笔记总结

    近日阅读Corelan Team编写的exploit writing tutorial系列,大致了解了一下原理,记了一些笔记.此系列文章有中文翻译版,在看雪论坛上发表. 英文版地址:https://w ...

  7. nginx 站点80跳443配置

    server { listen 80; server_name www.furhacker.cn; location /{# return 301; rewrite ^(.*)$ https://$h ...

  8. 快速使用shortcut,适配各种ROM

    地址(徐医生的GitHub):https://github.com/xuyisheng/ShortcutHelper 常用API     /**      * 添加快捷方式      *      * ...

  9. MediaPlayer 音频播放 示例

    状态机.流程图.生命周期 对播放音频/视频文件和流的控制是通过一个状态机来管理的.下图显示一个MediaPlayer对象被支持的播放控制操作驱动的生命周期和状态. 椭圆代表MediaPlayer对象可 ...

  10. Centos 5.2安装配置DNS服务器

    BIND安装配置(主从)我的系统环境:centos 5.2 作者:哈密瓜 主:我采用的是yum安装[root@linux src]#yum -y install bind* 生成rndc控制命令的ke ...