Tuesday 19 August 2008

观察者模式 observer 模拟监听器的实现

学过awt,尤其是swing的同学就知道,swing中observer模式被大量的使用。比如,button.addActionListener(...)后,一旦你点击button后就能触发相应的事件。很多同学一定想知道内部的机制。今天我正好也对这个议题感兴趣,那么我们一起来看一下。
假设我有一台电脑。电脑里面有一个事件,当你按下电源按钮的时候,那么显示器会亮,电源指示灯也会亮。
这是典型的一个类似于GUI中按钮事件模拟。
我们来分析一下需要几个类
电脑肯定是需要的,我们得出Computer类。Computer类内部有一个按电源按钮的动作,我们叫做clickButton.这个事件我们叫做EventPowerOn.于是得出了EventPowerOn类。另外,对比GUI里面的button,他们都会添加监听器,所以应该有个监听器类,我们叫做EventPowerOnListener。好到这里我们基本上找出了几个必须的类Computer,EventPowerOn ,EventPowerOnListener
我们写出他们的框架:
public class Computer implements Runnable{


public void clikButton(){}//该方法如果被调用,那么说明电源按钮被按下

}

public class EventPowerOn {
private Object source;//事件的来源,也就是时间是由哪一个对象激发的
private long time;//这个属性是随意添加的 比如社么时候电源按钮被按下

public EventPowerOn( long time, Object object) {
super();
this.source = object;
this.time = time;

}
public Object getObject() {
return source;
}
public void setObject(Object object) {
this.source = object;
}
public long getTime() {
return time;
}
public void setTime(long time) {
this.time = time;
}

}
public interface EventPowerOnListener {
public void performanceToPowerOn(EventPowerOn eventPowerOn);//改方法是电源按钮被
//按后应的响应
}
刚才我们说,电源按钮被按下,那么显示器和电源指示器会变亮。那么显示器和电源指示器应该都另外做一个类
所以有:
public class Screen implements EventPowerOnListener{

public void performanceToPowerOn(EventPowerOn eventPowerOn) {
System.out.println(eventPowerOn.getTime());


System.out.println("Screen is on");

}

}
public class Light implements EventPowerOnListener{
public void performanceToPowerOn(EventPowerOn eventPowerOn) {
System.out.println(eventPowerOn.getTime());


System.out.println("power light is on");

}
}
为什么我们要用他们实现监听器类呢,因为对computer来说,他们都是监听器。一旦电源按钮被按下,显示器和电源指示器就应该要有反应。
那么如何将screen 和light添加进computer 呢。我们做如下的改动
public class Computer {

private List listeners=new ArrayList();

public void addEventPowerOnListener(EventPowerOnListener eventPowerOnListener)
{
listeners.add(eventPowerOnListener);
}
public void clikButton(){
for(EventPowerOnListener temp:listeners)
{
temp.performanceToPowerOn(new EventPowerOn(System.currentTimeMillis(),this) );
}
}

}
好了,一个模拟的例子就完成了。我们来测试一下:
public class testObserver {
public static void main(String[] args){
Computer computer=new Computer();
Screen screen=new Screen();
Light light=new Light();
computer.addEventPowerOnListener(screen);
computer.addEventPowerOnListener(light);
computer.click();



};
}
看戏结果:
1219215665656
Screen is on
1219215665656
power light is on
说明clickButton,那么显示器和电源指示灯都会亮。我们还可以添加另外一些监听器,比如,硬盘会转等。。。只需实现listner接口以及添加到computer的监听列表即可。。。
(最近回过头来看java的一些基础的东西,看swing的时候,突然很想知道observer模式到底是则么实现的,研究了一下,看了一些源码,有了些心得,就顺便写出来)

No comments: