昨天开始看《深入浅出hibernate》,发现装饰模式看起来有点困难。今天稍稍理解了一下。
/**装饰,装饰真实对象的方法而已。实质就是利用子类重写父类的方法。
* 装饰对象和真实对象有相同的接口----接口中的方法都被装饰。真实对象不想装饰的方法不要放在 接口中。
* 装饰抽象类的作用是:不同的派生类,可以给真实对象需要装饰的方法增加不同的功能。
* (1) 装饰对象和真实对象有相同的接口。这样客户端对象就可以和真实对象相同的方式和装饰对象交互。
(2) 装饰对象包含一个真实对象的引用(reference)
(3) 装饰对象接受所有来自客户端的请求。它把这些请求转发给真实的对象。
(4) 装饰对象可以在转发这些请求以前或以后增加一些附加功能。这样就确保了在运行时,不用修改给定对象的结构就可以在外部增加附加的功能。在面向对象的设计中,通常是通过继承来实现对给定类的功能扩展。
*/
package test;
/**接口-目的是提取实现类中所有需要装饰的方法
* 方法包括了实现类中所有需要装饰的方法
* @author yutian
*
*/
interface Icom{
public void show(String msg);
public void say(String msg);
}
//接口实现的类----装饰的真实对象
class com_cmp implements Icom{
/**非装饰的方法
* @param msg
*/
public void print(String msg) {
// TODO Auto-generated method stub
System.out.println(msg);
}
@Override
public void show(String msg) {
// TODO Auto-generated method stub
System.out.println(msg);
}
@Override
public void say(String msg) {
// TODO Auto-generated method stub
System.out.println(msg);
}
}
/**装饰抽象类---对接接口,实现接口的所有方法
* 先产生它的原因是:不同的派生类,可以给真实对象需要装饰的方法增加不同的功能。
* @author yutian
*
*/
class ab_decor_cmp implements Icom{
private Icom icom;
/**
* @param icom 真实对象,即需要增加功能的对象
*/
public void setIcom(Icom icom) {
this.icom = icom;
}
/* 调用真实的对象完成功能----不增不减,方便不同的派生类,可以给真实对象增加不同的功能。
* @see test.Icom#show(java.lang.String)
*/
@Override
public void show(String msg) {
// TODO Auto-generated method stub
if(icom!=null)
icom.show(msg);
}
@Override
public void say(String msg) {
// TODO Auto-generated method stub
if(icom!=null)
icom.say(msg);
}
}
//装饰具体类,给真实对象需要装饰的方法增加的功能。
class decor_cmp extends ab_decor_cmp{
public void show(String msg) {
// TODO Auto-generated method stub
super.show(msg);
add_oper();
}
@Override
public void say(String msg) {
// TODO Auto-generated method stub
super.say(msg);
add_oper();
}
private void add_oper(){
System.out.println("new operation!");
}
}
public class Decorator {
public static void main(String[] args){
Icom cmp = new com_cmp();//接口类型的cmp隔离了com_cmp中不想装饰的方法
decor_cmp dec = new decor_cmp();
dec.setIcom(cmp);//必须先调用父类的set方法,将真实对象加进装饰对象 <pre name="code" class="html"> <span style="white-space:pre"> </span>dec.show("show");dec.say("say");}}
/** * 调用的真实对象com_cmp的方法,但增加了新的功能 */
<span style="white-space:pre"> </span>}
}