设计模式4:装饰模式(1)

  作用:

  基于已经存在的功能提供增强功能。

  将扩展的类作为新的类的构造函数参数传入,然后对这个类进行装饰处理。

  为了让装饰者有被装饰的功能,需要存在继承关系。

  简单的点讲就是为了扩展一个已定义好的类的功能,而不修改原来类的定义,原来的类仍然可以继续使用,现在定义一个新的类继承原来的类,同时在这个新的类中定义父类的成员变量,同时在构造方法中将外界的一个父类对象传给这个成员变量,然后既可以通过这个成员变量使用父类的方法,同时又可以增加新的方法实现对父类功能的扩充。例如IO流中很多类都采用了装饰模式,如BufferedReader类是装饰了Reader类,LineNumberReader类装饰了BufferedReader类。

  示例1:

 1 package com.qianfeng.decorator;
 2 
 3 public class Demo12 {
 4 
 5     public static void main(String[] args) {
 6         Person p1 = new Zhansan();
 7         p1.show();
 8 
 9         Person p2 = new JeansDecorator(p1);
10         p2.show();
11 
12         Person p3 = new ShirtDecorator(p1);
13         p3.show();
14     }
15 }
16 
17 interface Person {
18     void show();// 显示一个人
19 }
20 
21 abstract class DecoratorPerson implements Person {
22     private Person person;
23 
24     DecoratorPerson(Person person) {
25         this.person = person;
26     }
27 }
28 
29 class JeansDecorator extends DecoratorPerson {
30 
31     JeansDecorator(Person person) {
32         super(person);
33         // TODO Auto-generated constructor stub
34     }
35 
36     @Override
37     public void show() {
38         System.out.println("穿牛仔裤的person");
39 
40     }
41 }
42 
43 class ShirtDecorator extends DecoratorPerson {
44 
45     ShirtDecorator(Person person) {
46         super(person);
47         // TODO Auto-generated constructor stub
48     }
49 
50     @Override
51     public void show() {
52         System.out.println("穿t恤的person");
53 
54     }
55 
56 }
57 
58 class Zhansan implements Person {
59 
60     @Override
61     public void show() {
62         System.out.println("没有任何装饰的张三");
63 
64     }
65 
66 }

 

  示例2:

  1 package com.qianfeng.decorator;
  2 
  3 import java.io.FileReader;
  4 import java.io.IOException;
  5 import java.io.Reader;
  6 
  7 public class Demo13 {
  8     // LineNumberReader 装饰类,继承了BufferedReader
  9     public static void main(String[] args) throws IOException {
 10         FileReader fr = new FileReader("src/temp.txt");
 11         // LineNumberReader lnr = new LineNumberReader(fr);
 12         // String line = null;
 13         // lnr.setLineNumber(100);
 14         // while ((line = lnr.readLine()) != null) {
 15         // System.out.println(lnr.getLineNumber() + ":" + line);
 16         // }
 17         // lnr.close();
 18 
 19         MyLineNumberReader lnr = new MyLineNumberReader(fr);
 20         String line = null;
 21         // lnr.setLineNumber(1000);
 22         while ((line = lnr.MyReadLine()) != null) {
 23             System.out.println(lnr.getLineNumber() + ":" + line);
 24         }
 25         lnr.myClose();
 26     }
 27 
 28 }
 29 
 30 /*
 31  * setLineNumber():设置起始行号
 32  * 
 33  * myReadLine():每次读取一行
 34  * 
 35  * getLineNumber():获得当前行号
 36  */
 37 class MyBufferedReader {
 38     private Reader r; // 真正读取功能的类
 39     private char[] arr = new char[512];// 相当于缓冲区
 40     private int index; // 数组下标
 41     private int count; // 统计缓冲区中字符个数
 42 
 43     public MyBufferedReader(Reader r) {
 44         this.r = r;
 45     }
 46 
 47     // 实现一次读取一个的功能
 48     public int myRead() throws IOException {
 49         // 缓冲区中是否有数据
 50         if (count == 0) {
 51             // 从文件中读取数据到缓冲区,返回值读取的字符数
 52             count = r.read(arr);
 53             index = 0; // 下标为0
 54         }
 55         if (count < 0) // 文件末尾
 56             return -1;
 57         // 从缓冲区中读取一个字符
 58         int num = arr[index];
 59         index++;// 下标+1
 60         // 数量-1
 61         count--;
 62         return num;
 63 
 64     }
 65 
 66     // 一次读取一行
 67     public String myReadLine() throws IOException {
 68         StringBuilder sb = new StringBuilder();
 69         int num;
 70         while ((num = myRead()) != -1) {
 71             if (num == '\r')
 72                 continue;
 73             else if (num == '\n')
 74                 return sb.toString();
 75             else
 76                 sb.append((char) num);
 77         }
 78         return null;
 79     }
 80 
 81     // 关闭流
 82     public void myClose() throws IOException {
 83         r.close();
 84     }
 85 }
 86 
 87 class MyLineNumberReader extends MyBufferedReader {
 88 
 89     private int lineNumber;
 90 
 91     public MyLineNumberReader(Reader r) {
 92         super(r);
 93     }
 94 
 95     public void setLineNumber(int lineNumber) {
 96         this.lineNumber = lineNumber;
 97     }
 98 
 99     public int getLineNumber() {
100         return this.lineNumber;
101     }
102 
103     // 每次读取一行,行号+1
104     public String MyReadLine() throws IOException {
105         ++lineNumber;
106         return super.myReadLine();
107     }
108 
109 }

  

  未完,待续。

 

 

  

  

posted on 2015-09-12 21:07  AellenLei  阅读(181)  评论(0编辑  收藏  举报

导航