读书人

装饰模式的懂得

发布时间: 2012-10-26 10:30:59 作者: rapoo

装饰模式的理解

目前有这么个场景:三个继承关系Father,Son,GrandSon三个类,我要再Son类上增强一些功能怎么办?我想项目过程中,如果你是后来者,那么你免不了修改别人的代码。如果初期你对那些代码不知所云,但目前又急需修改功能,难道你真决定在Son类上大动手脚,我想你没那么牛叉,至少我对那些功能点的修改就有点畏首畏尾,毕竟前期代码不是我写,稍微修改,意想不到的错误无法预料。不允许,对了,为什么呢?你增强的功能是修改Son类中的方法吗?增加方法吗 ?对GrandSon的影响哪?如果GrandSon有多个的情况,你怎么办?这个评估的工作量就是够你受的,所以这个是不允许的,那还是要解决问题的呀,怎么办?这里就可以用着装饰模式了。

装饰模式主要用于对已有功能动态地添加功能的一种方式。通过对对象进行包装,使得装饰对象的实现和对象进行分离,从而装饰对象只需要关心自己的功能,而不需要关心如何添加到对象链中。这样做就是为了有效地把类的核心职责和装饰功能区分开来,但要注意地是,装饰模式的顺序非常重要。下面就用代码对上面场景进行模拟:

/**

?* @author lw 父亲类方法有a 、b 、c三种方法,其中c是抽象方法。

?*/

public abstract class Father {

??? public void a() {

?????? System.out.println("Father a函数!");

??? };

??? public void b() {

?????? System.out.println("Father b函数!");

??? };

??? public abstract void c();

/**

?* @author lw

?* 儿子继承父亲,重写父亲的抽象方法c

?*/

public class Son1 extends Father {

?

??? @Override

??? public void c() {

?????? System.out.println("Son1 C函数重写!");

??? }

?

??? @Override

??? public void b() {?? //需要对b的方法进行扩展

?????? System.out.println("overrideSon1's b function()");

??? }

在这里GrandSon1的代码就不做模拟,假设Son1的b方法进行修改,将影响到其子类的功能实现。此时需要对Son1的b方法功能进行扩展,其中扩展的功能代码是核心代码,所以我把它先拿到外面搁置起来,并对其进行装饰。

public abstract class Decorator extends Father{

?

??? private Father father;

???

??? protected Decorator(Father father){

?????? this.father=father;

??? }

???

??? public void b(){

?????? if(father!=null){

?????????? father.b();

?????? }

??? }

???

??? @Override

??? public void c() {

?????? System.out.println("重写C函数");

??? }

public? class Decorator1Impl extends Decorator{

??? public Decorator1Impl(Father father){

?????? super(father);

??? }

??? public void b(){

?????? super.b();

?????? System.out.println("装饰模式下的b 1方法");

??? }

public? class Decorator2Impl extends Decorator{

??? public Decorator2Impl(Father father){

?????? super(father);

??? }

???

??? public void b(){

?????? super.b();

?????? System.out.println("装饰模式下的b 2方法");

??? }

public class Client {

??? public static void main(String[] args) {

?????? Father f;

?????? f = new Son1();

?????? Father f1=new Decorator1Impl(f);

?????? Father f2=new Decorator2Impl(f1);

?????? f2.b();

??? }

override Son1's b function()

装饰模式下的b 1方法

如果某一天,你觉得这个调用顺序需要更改,到时只需要更改装饰模式下的接口实现就能完成程序的扩展。或者某天你需要继续对该功能进行扩充,到时只需要继续写接口实现类即可。但是对于装饰模式千万要注意其顺序。


?

装饰模式是对继承的有力补充。在项目开发实施过程中,将要考虑系统的易维护、易扩展、易复用等情况,但继承是静态地给类增加功能,如果用继承解决的话,将会增加很多类,灵活性也会变差。而装饰模式有非常好的优点,扩展性比较好。在系统开发过程中,需求的变更我想这是永远也改变不了的现实,装饰模式可以提供很好的帮助。通过重新封装一个类,而不是通过继承去实现,这样对原程序没有变更,但通过扩充很好地完成了功能的增强。

读书人网 >软件架构设计

热点推荐