编程爱好者联盟 2016-12-08
装饰:基于已有的功能,并提供加强功能,装饰类通常会通过构造方法接收被装饰的对象。并基于被装饰的对象的功能,提供更强的功能。相对于继承来说装饰模式可以在不使用创造更多子类的情况下,将对象的功能加以扩展。英文叫:Decorator
继承:因为要扩展某个功能,就得继承父类,然后覆盖重写,导致这个体系更加臃肿,并且内存占用高。
一般情况下,其实接口也可以当做一种抽象类来看待,也就是说父类对象也可以是接口,如果一个类A实现了一个接口T,当我们在声明实例:T t =new A();时,t看起来是一个T,本质上是一个A,这也是多态的一种!
验证一下,首先先创建一个接口a:
public class test {
interface a{
void fun();
}
class b implements a{
public void fun() {
System.out.println("这个实现接口类!");
}
}
class c extends b{
public void fun(){
System.out.println("这是实现接口类的子类!!");
}
}
public static void main(String[] args) {
a dd = new test().new c();
dd.fun();
}
} 运行:

摘自CSDN的一位大神的例子,看这个程序只要多态的功能理清楚也就明白了,说白了装饰模式无非就是多态与构造方法的灵活运用罢了:
public class decorateMode{
//定义被装饰者
public interface Human {
public void wearClothes();
public void walkToWhere();
}
//定义装饰者
public abstract class Decorator implements Human {
private Human human;
public Decorator(Human human) {
this.human = human;
}
public void wearClothes() {
human.wearClothes();
}
public void walkToWhere() {
human.walkToWhere();
}
}
//下面定义三种装饰,这是第一个,第二个第三个功能依次细化,即装饰者的功能越来越多
public class Decorator_zero extends Decorator {
public Decorator_zero(Human human) {
super(human);
}
public void goHome() {
System.out.println("进房子。。");
}
public void findMap() {
System.out.println("书房找找Map。。");
}
public void wearClothes() {
// TODO Auto-generated method stub
super.wearClothes();
goHome();
}
public void walkToWhere() {
// TODO Auto-generated method stub
super.walkToWhere();
findMap();
}
}
public class Decorator_first extends Decorator {
public Decorator_first(Human human) {
super(human);
}
public void goClothespress() {
System.out.println("去衣柜找找看。。");
}
public void findPlaceOnMap() {
System.out.println("在Map上找找。。");
}
public void wearClothes() {
// TODO Auto-generated method stub
super.wearClothes();
goClothespress();
}
public void walkToWhere() {
// TODO Auto-generated method stub
super.walkToWhere();
findPlaceOnMap();
}
}
public class Decorator_two extends Decorator {
public Decorator_two(Human human) {
super(human);
}
public void findClothes() {
System.out.println("找到一件D&G。。");
}
public void findTheTarget() {
System.out.println("在Map上找到神秘花园和城堡。。");
}
public void wearClothes() {
// TODO Auto-generated method stub
super.wearClothes();
findClothes();
}
public void walkToWhere() {
// TODO Auto-generated method stub
super.walkToWhere();
findTheTarget();
}
}
//定义被装饰者,被装饰者初始状态有些自己的装饰
public class Person implements Human {
public void wearClothes() {
// TODO Auto-generated method stub
System.out.println("穿什么呢。。");
}
public void walkToWhere() {
// TODO Auto-generated method stub
System.out.println("去哪里呢。。");
}
}
//测试类,看一下你就会发现,跟java的I/O操作有多么相似
public static void main(String[] args) {
decorateMode decorateMode=new decorateMode();
Human person =decorateMode. new Person();
Decorator decorator = decorateMode.new Decorator_two(decorateMode.new Decorator_first(
decorateMode.new Decorator_zero(person)));
decorator.wearClothes();
decorator.walkToWhere();
}
} 运行结果:

读取文件的行内容,并给每行前加上行数、行后加上分号
package com.beiwo.Io;
import java.io.BufferedReader;
import java.io.FileNotFoundException;
import java.io.FileReader;
import java.io.IOException;
import java.io.Reader;
/*
BufferedReader : 拓展FileReader的功能。
需求1 :实现通过readLine读取代码 ,每一行加上一个行号。
需求2 :实现通过readLine读取代码 ,每一行加上一个分号。
需求3 :实现通过readLine读取代码 ,每一行加上一个引号。
//=========================================================
需求4 :实现通过readLine读取代码 ,每一行加上一个引号 + 行号。
需求5 :实现通过readLine读取代码 ,每一行加上一个分号 + 行号。
需求6 :实现通过readLine读取代码 ,每一行加上一个分号 + 引号。
需求7 :实现通过readLine读取代码 ,每一行加上一个分号 + 引号 + 行号。
如果用继承:
好处:代码结构和清晰容易理解
缺点:继承体系会很庞大。实现七个需求就得创建七个子类
装饰者模式 : 增强一个类的功能还可以让装饰者类之间互相装饰。
*/
//添加行号
class BufferedReaderLineNum extends BufferedReader{
//定义一个对象来接收传进来的对象
BufferedReader bufferedReader;
int count = 1;
//子类继承父类会默认调用父类的无参构造方法
public BufferedReaderLineNum (BufferedReader in) {
super(in);
this.bufferedReader = in;
}
//复写readLine方法
@Override
public String readLine() throws IOException {
// TODO Auto-generated method stub
//调用父类的readLine
String content = bufferedReader.readLine();
//给内容田间一个行号
if(content == null){
return null;
}
content = count+" "+content;
count++;
return content;
}
}
//添加分号的
class BufferedReaderLineSemi extends BufferedReader{
//定义一个对象来接收传进来的对象
BufferedReader bufferedReader;
//子类继承父类会默认调用父类的无参构造方法
public BufferedReaderLineSemi (BufferedReader in) {
super(in);
this.bufferedReader = in;
}
//复写(重写)readLine方法
@Override
public String readLine() throws IOException {
// TODO Auto-generated method stub
//调用父类的readLine
String content = bufferedReader.readLine();
//给内容田间一个行号
if(content == null){//表示数据已经读完
return null;
}
return content+";";
}
}
public class Demo1 {
/**
* @param args
* @throws IOException
*/
public static void main(String[] args) throws IOException {
// TODO Auto-generated method stub
testLineNum();
}
public static void testLineNum () throws IOException{
//1.开启一个通道,并且带一个文件路径
FileReader reader = new FileReader("这里填上文件路径如:C:\\a.java");
//创建缓冲流
BufferedReader bufferedReader = new BufferedReader(reader);
//2.创建一个带分号的缓冲流
BufferedReaderLineSemi lineSemi = new BufferedReaderLineSemi(bufferedReader);
//创建一个缓冲流 ,带行号加分号
BufferedReaderLineNum lineNum = new BufferedReaderLineNum(lineSemi);
//3.开始读取数据
String content = null;
while((content = lineNum.readLine())!=null){
System.out.println(content);
}
//4.关闭资源
lineNum.close();
}
} 效果:
