装饰模式(decorator pattern)

依照Num模型。讨论职业/IProfession类层次。

IProfession定义了方法say(String),事实上现类教师/ Teacher、医生、律师……依照其职业习惯给出say(String)的实现,这些实现类相当于Num模型中的Zero。

而才艺/Talent类层次(相当于NextOne),能够装饰基本对象,say(String)实现中,在必要的时候会说english、在必要的时候会唱几句……


package closure.decorator;
public interface IProfession {
    abstract public void say(String s);
}

package closure.decorator;
import static yqj2065.util.Print.pln;
public class Teacher implements IProfession{ 
    @Override public void say(String s){   pln(teach(s)) ;    }
    public String teach(String s){
        String myStyle="解说["+s+"]";
        return myStyle;
    }
}

package closure.decorator;
public abstract class Talent implements IProfession {
    protected IProfession base;
    public Talent(IProfession base) {   this.base = base;    }
}  
package closure.decorator;
public class TSong extends Talent{
   public TSong(IProfession base) {
       super(base);
    }
    @Override public void say(String s){
        s = this.song(s);        
        base.say(s); //
    }    
    public String song(String s){
        String strengthen="旋律("+s+")";
        return strengthen;
    }
}// TEnglish略


1.模板方法

抽象装饰类型Talent,只定义了其类层次的结构——作为类型闭包。没有提供say(String s)的默认实现,由于提供例如以下默认实现

    @Override public final void say(String s) {

        base.say(s);

    }

尽管暗示其子类型应该调用base.say(),但不具有强制性。

Talent能够编写模板方法,使得详细装饰类不再须要改写say()并调用base.say(),而只关注编写增强say()功能的代码。

package closure.decorator;
public abstract class Talent implements IProfession {
    protected IProfession base;
    public Talent(IProfession base) {   this.base = base;    }
    @Override public final void say(String s) { //模板方法
        s = strengthen(s);
        base.say(s);
    }
    public abstract String strengthen(String s);
}
package closure.decorator;
public class TEnglish extends Talent {
public TEnglish(IProfession base) {        super(base);    }
public String english(String s) {
        String strengthen ="E文(" + s + ")";
        return strengthen;
    }
    @Override  public String strengthen(String s) {       return this.english(s);    }
}
//Test
        IProfession p = (IProfession) God.create("Profession-decorator");//
        p.say("类层次");        
        p = new TEnglish(new TSong(new TEnglish(p)));
        p.say("LSP");
        p = new TSong(p);
        p.say("继承");
        p= p.removeSong();
        p.say("继承");

測试代码的输出为:

解说[类层次]

解说[E文(旋律(E文(LSP)))]

解说[E文(旋律(E文(旋律(继承))))]

装饰模式比較典型的标志:Person b = new T2(newT2(newT1(newLawyer())));

2.装饰模式与桥接模式

装饰模式与桥接模式非常相似。桥接模式将2次以上的策略选择逐一地串接起来。如例程3-5中定义的模板方法。

    public default void lecture(String s) {

        String myStyle = say(s);

        next.say(myStyle);

    }

桥接模式所串接的,是两种不同的类层次。而装饰模式则将Talent和它的父类型IProfession进行逐一的串接。

能够说,装饰模式是桥接模式的特例。




posted on 2017-08-07 09:14  yutingliuyl  阅读(140)  评论(0编辑  收藏  举报