参考文档:http://lavasoft.blog.51cto.com/62575/90853/

1.当我们需要为某个现有的对象,动态的增加一个新的功能或职责时,可以考虑使用装饰模式。

2.当某个对象的职责经常发生变化或者经常需要动态的增加职责,避免为了适应这样的变化,而增加继承子类扩展的方式,因为这种方式会造成子类膨胀的速度过快,难以控制。

思路:将一个类 通过装饰模式扩展他的功能。1.实际完成功能的类 2.装饰类 他们继承同一个接口, 因此两个类都会实现同一个方法。装饰类持有 实际类对象的引用,在实现共同方法时,调用的是 实际类的实现,在调用实际方法的实现时,可以在上下进行一定的扩展。

一、装饰模式使用场景:

在不修改原有方法的基础上,对原方法的功能进行扩展追加一些功能

二、核心思想:

对接口方法进行包装。

1.被装饰对象和装饰对象实现相同的接口。

2.装饰类持有被装饰对象,接口方法的实际实现通过调用:被装饰对象的接口方法。

3.在调用被装饰对象的接口方法前可以增加其他的方法。

示例代码:转载自参考文献http://lavasoft.blog.51cto.com/62575/90853/

1.定义一个目标接口

/** * Created by IntelliJ IDEA. * User: leizhimin * Date: 2008-8-3 12:51:06 * 项目 */
public interface Project {
/** * 写代码 */
void doCoding();
}

2.要被装饰的对象和负责装饰的对象都实现该接口

要被装饰的对象

/** * Created by IntelliJ IDEA. * User: leizhimin * Date: 2008-8-3 12:52:12 * 代码工人 */
public class Employe implements Project{ /** * 编码 */
public void doCoding(){
System.out.println("代码工人 在编写代码,加班编啊编啊,终于编完了!"
);
}
}

负责装饰的对象

/** * Created by IntelliJ IDEA. * User: leizhimin * Date: 2008-8-3 12:51:26 * 项目经理 */
public class Manager implements Project {
private Project project; //实际上存放的是代码工人对象
public Manager(Project project) {
this.project = project;
} /** * 编码 */
public void doCoding() { //项目经理开始新的工作
startNewWork(); } /** * 模板:定义项目经理自己的事情 */
public void startNewWork() { //项目经理在做早期工作
doEarlyWork(); //项目经理很牛,做完需求和设计后,直接将编码委派给代码工人干
project.doCoding(); //项目经理在做收尾工作
doEndWork();
} /** * 项目经理自己的事情:做早期工作 */
public void doEarlyWork() { } /** * 项目经理做收尾工作 */
public void doEndWork() { }
}

对Manager的类进行一定的扩展

/** * Created by IntelliJ IDEA. * User: leizhimin * Date: 2008-8-3 13:45:18 * 具体的项目经理A */
public class ManagerA extends Manager{
public ManagerA(Project project) {
super(project);
}
/*** 项目经理自己的事情:做早期工作 */
public void doEarlyWork() {
System.out.println("项目经理A 在做需求分析");
System.out.println("项目经理A 在做架构设计");
System.out.println("项目经理A 在做详细设计");
}
}

注释:可以看到负责装饰的对象对接口方法的实现功能实际上是由 project.doCoding来完成的。在调用project.doCoding方法前后,都进行了扩展。

通过调用对Project进行扩展的装饰类来实现doCoding:

/** * Created by IntelliJ IDEA. * User: leizhimin * Date: 2008-8-3 13:03:22 * 客户端测试 */
public class Client {
public static void main(String args[]) {
Project employe = new Employe(); //代码工人
Project managerA = newManagerA(employe); //项目经理
Project managerB = new ManagerB(employe); //项目经理
//以经理的名义将编码完成,功劳都是经理的,实际编码的是工人
managerA.doCoding();
managerB.doCoding();
}
}

Employe 传递给Manager进行装饰。

Manager 装饰后再次实现doCoding