代理模式 vs 装饰模式
代理模式和装饰模式有很大的相似性,二者的类图(几乎)是一样的。下面分别讲解代理模式和装饰模式。
1、代理模式
一般著名的跑步运动员都会有自己的代理人,如果想联系该运动员的比赛事宜,可以直接联系他的代理人就可以了。类图如下所示:
IRunner接口如下:
public interface IRunner { public void run(); }
Runner类如下所示:
public class Runner implements IRunner { @Override public void run() { System.out.println("运动员在跑步..."); } }
RunnerAgent代理类如下:
public class RunnerAgent implements IRunner { private IRunner runner; public RunnerAgent(IRunner runner) { this.runner = runner; } @Override public void run() { Random rand = new Random(); if (rand.nextBoolean()) { System.out.println("代理人安排运动员跑步..."); runner.run(); } else { System.out.println("代理人有事情,不安排运动员跑步了..."); } } }
测试场景如下:
public class Main { public static void main(String[] args) { IRunner runner = new Runner(); IRunner agent = new RunnerAgent(runner); System.out.println("有人请求代理人让运动员跑步..."); agent.run(); } }
输出结果为:
或者
2、装饰模式
面对代理模式中的场景,使用装饰模式如何实现呢?装饰模式是对类功能的加强,比如增强跑步速度,安装一个动力装置等。类图如下所示:
是不是和代理模式的类图一模一样,其实是一样的,但是实现的意图是一样的,先看代码:
public class RunnerWithJet implements IRunner { private IRunner runner; public RunnerWithJet(IRunner runner) { this.runner = runner; } @Override public void run() { System.out.println("给运动员屁股后加一个推进装置..."); runner.run(); } }
测试场景如下:
public class Main { public static void main(String[] args) { IRunner runner = new Runner(); IRunner superRunner = new RunnerWithJet(runner); System.out.println("有人请求加强版运动员跑步..."); superRunner.run(); } }
输出结果如下:
3、二者的比较
通过上述的例子可以看出,代理模式和装饰模式非常类似,甚至代码都类似。二者最主要的区别是:代理模式中,代理类对被代理的对象有控制权,决定其执行或者不执行。而装饰模式中,装饰类对代理对象没有控制权,只能为其增加一层装饰,以加强被装饰对象的功能,仅此而已。
代理模式使用到极致开发就是AOP, 这是各位采用Spring架构开发必然要使用到的技术(Spring学习之第一个AOP程序),它就是使用了代理和反射的技术。代理模式在Java的开发中俯拾皆是, 是大家非常熟悉的模式, 应用非常广泛, 而装饰模式是一个比较拘谨的模式, 在实际应用中接触比较少, 但是也有不少框架项目使用了装饰模式, 例如在JDK的java.io.*包中就大量使用装饰模式, 类似如下的代码:
OutputStream out = new DataOutputStream( new FileOutputStream( "test.txt") )
这是装饰模式的一个典型应用, 使用DataOutputStream封装了一个FileOutputStream, 以方便进行输出流处理。
参考资料:
1、《设计模式之禅》机构类模式大PK章节