每周写工作总结、工作汇报是许多程序员的伤痛,哈哈。真实情况是又摸鱼了一周,给领导写工作报告,当然不能直接写“摸鱼”,这时候就需要装饰模式来对我们提交的报告做下修饰,甚至领导的评语里面也给加点美化内容,不然, 这个月的奖金就泡汤了,只能吃土了。
一、定义
定义(百度百科):装饰模式指的是在不必改变原类文件和使用继承的情况下,动态地扩展一个对象的功能。它是通过创建一个包装对象(装饰对象)来包裹真实的对象。
重点:通过创建包装类,包裹真实对象,达到对真实对象的动态扩展。
二、UML
看下我们上面的场景用装饰模式实现的UML图。
1、WorkReport为工作报告的抽象类,定义了两个方法,reportdata()用于查看工作报告的内容,check方法是领导点评的。
2、RealReport为实际的报告类。
3、Decorator为装饰类的抽象类,经过装饰后的报告依然得是报告,所以装饰抽象类继承自工作报告
4、DecAddDescript、DecAddComment为具体的装饰类。
三、代码实例
看下代码实现
1、工作报告抽象类
package com.design.decorator;
public abstract class WorkReport {
public abstract void reportData();
public abstract void check(String comment);
}
2、工作报告
package com.design.decorator;
public class RealReport extends WorkReport {
@Override
public void reportData() {
System.out.println("工作描述:");
System.out.println("这周的工作是调试了俩个BUG!!");
}
@Override
public void check(String comment) {
System.out.println("评述:");
System.out.println(comment);
}
}
3、装饰抽象类,持有工作报告的引用,扩展报告。
package com.design.decorator;
public abstract class Decorator extends WorkReport {
protected WorkReport workReport;
public Decorator(WorkReport workReport){
this.workReport = workReport;
}
}
4、具体的装饰类,扩展查看报告方法
package com.design.decorator;
public class DecAddDescript extends Decorator{
public DecAddDescript(WorkReport workReport) {
super(workReport);
}
@Override
public void reportData() {
//动态扩展,增加一些描述
super.workReport.reportData();
System.out.println("这周出来的两个BUG比较诡异!经过各种折腾,费尽九牛二虎之类才搞定的,别看只是两个bug,工作量可不少!!");
System.out.println();
}
@Override
public void check(String comment) {
super.workReport.check(comment);
}
}
具体的装饰类,扩展领导审核功能
package com.design.decorator;
public class DecAddComment extends Decorator {
public DecAddComment(WorkReport workReport) {
super(workReport);
}
@Override
public void reportData() {
workReport.reportData();
}
@Override
public void check(String comment) {
workReport.check(comment);
////动态扩展,悄悄把领导的评语后面增加一些内容
System.out.println("XXX同学虽然只改两个BUG,但修复这两个BUG意义重大,给他全额奖金!!");
}
}
5、最终运行
package com.design.decorator;
public class TestMain {
public static void main(String[] args) {
WorkReport workReport = new RealReport();
//装饰一下描述
WorkReport decAddDescript = new DecAddDescript(workReport);
//装饰一下领导评语,背着领导偷偷加
WorkReport decACommnet = new DecAddComment(decAddDescript);
decACommnet.reportData();
decACommnet.check("修复了两个bug!!");
}
}
测试结果
四、总结
从定义中,我们就可以知道装饰模式的用处了,扩展对象的功能,而且是在不修改原对象的情况下。
优点:
1、有力于分离开核心功能与非核心的扩展功能。我们可以将核心功能放到实际类上,其他非核心装饰类的功能放到装饰类上。
2、灵活扩展装饰,如上例子,如果我们不需要扩展评语,只扩展工作描述,则只需要在装饰链上面,去除评语对象的。只需修改客户端代码即可。新增扩展功能,也仅需增加装饰类。
WorkReport workReport = new RealReport();
//装饰一下描述
WorkReport decAddDescript = new DecAddDescript(workReport);
缺点:
1、装饰链较长情况下,逻辑比较复杂,不利于阅读代码。