设计模式-观察者模式

/*
* 观察者模式
* 定义对象间的一种一个(Subject)对多(Observer)的依赖关系,当一个对象的状态发送改变时,所以依赖于它的
* 对象都得到通知并被自动更新
*
* 当然,MVC只是Observer模式的一个实例。Observer模式要解决的问题为:
* 建立一个一(Subject)对多(Observer)的依赖关系,并且做到当“一”变化的时候,
* 依赖这个“一”的多也能够同步改变。最常见的一个例子就是:对同一组数据进行统计分析时候,
* 我们希望能够提供多种形式的表示(例如以表格进行统计显示、柱状图统计显示、百分比统计显示等)。
* 这些表示都依赖于同一组数据,我们当然需要当数据改变的时候,所有的统计的显示都能够同时改变。
* Observer模式就是解决了这一个问题。
*
* 适用性:
* 1. 当一个抽象模型有两个方面,其中一个方面依赖于另一方面
* 将这两者封装成独立的对象中以使它们可以各自独立的改变和服用
*
* 2. 当对一个对象的改变需要同时改变其他对象,而不知道具体有多少对象有待改变
*
* 3. 当一个对象必须通知其它对象,而它又不能假定其它对象是谁
*
* 参与者:
* 1. Subject(目标)
* 目标知道它的观察者,可以有任意多个观察者观察同一个目标
* 提供注册和删除观察者对象的接口
*
* 2. Observer(观察者)
* 为那些在目标发生改变时需获得通知的对象定义个更新的接口
*
* 3. ConcreteSubject(具体目标)
* 将有关状态存入各ConcreteObserver对象
* 当它的状态发送改变时,向它的各个观察者发出通知
*
* 4. ConcreteObserver(具体观察者)
* 维护一个指向ConcreteObserver对象的引用
* 存储有关状态,这些状态应与目标的状态保持一致
* 实现Observer的更新接口是自身状态与目标的状态保持一致
*
*
* */
*

实现:男生是观察者,女生是被观察者,当女生心情好的时候,会被男生观察到,然后采取把妹措施;

public abstract class Boy {//抽象观察者

    public abstract void action(String str);


}

具体观察者:

public class HandsomeBoy extends Boy{

    @Override
    public void action(String str) {

        System.out.println("girl "+str+" 我是帅哥,我要跟她约会去");
    }

}
public class UglyBoy extends Boy{

    @Override
    public void action(String str) {
        System.out.println("girl "+str+" 我是丑男,我要约她吃饭");

    }

}

抽象主题角色

public interface Girl{//被观察者

    public void addWatcher(Boy boy);

    public void removeWatcher(Boy boy);

    public void notifyWatchers(String str);
}

具体主题角色

public class BeautifulGirl implements Girl {

    private List<Boy> boysList = new ArrayList<Boy>();

    @Override
    public void addWatcher(Boy boy) {
        if (boy == null) {
            throw new IllegalArgumentException("the boy is null");
        }
        synchronized (boysList) {
            if (boysList.contains(boy)) {
                throw new IllegalArgumentException("the boy " + boy + " has been registered");
            }
            boysList.add(boy);
         }
    }

    @Override
    public void removeWatcher(Boy boy) {
        if (boy == null) {
            throw new IllegalArgumentException("the boy is null");
        }
        synchronized (boysList) {
            int index = boysList.indexOf(boy);
            if (index == -1) {
                throw new IllegalArgumentException("the boy " + boy + " was not registered");
            }
            boysList.remove(index);
         }
    }

    @Override
    public void notifyWatchers(String str) {
        synchronized (boysList) {
            for (Boy boy : boysList) {
                boy.action(str);
            }
        }

    }

}

测试:

public class Test {
public static void main(String[] args) {
    Girl girl = new BeautifulGirl();
    girl.addWatcher(new HandsomeBoy());
    girl.addWatcher(new UglyBoy());
    girl.notifyWatchers("我今天有时间");
}
}

输出:
girl 我今天有时间 我是帅哥,我要跟她约会去
girl 我今天有时间 我是丑男,我要约她吃饭

观察者模式java已经做了封装,

如果要想实现观察者模式,则必须依靠java.util包中提供的Observable类和Observer接口。

import java.util.* ;
class House extends Observable{ // 表示房子可以被观察
private float price ;// 价钱
public House(float price){
this.price = price ;
}
public float getPrice(){
return this.price ;
}
public void setPrice(float price){
// 每一次修改的时候都应该引起观察者的注意
super.setChanged() ; // 设置变化点
super.notifyObservers(price) ;// 价格被改变
this.price = price ;
}
public String toString(){
return “房子价格为:” + this.price ;
}
};
class HousePriceObserver implements Observer{
private String name ;
public HousePriceObserver(String name){ // 设置每一个购房者的名字
this.name = name ;
}
public void update(Observable o,Object arg){
if(arg instanceof Float){
System.out.print(this.name + “观察到价格更改为:”) ;
System.out.println(((Float)arg).floatValue()) ;
}
}
};
public class ObserDemo01{
public static void main(String args[]){
House h = new House(1000000) ;
HousePriceObserver hpo1 = new HousePriceObserver(“购房者A”) ;
HousePriceObserver hpo2 = new HousePriceObserver(“购房者B”) ;
HousePriceObserver hpo3 = new HousePriceObserver(“购房者C”) ;
h.addObserver(hpo1) ;
h.addObserver(hpo2) ;
h.addObserver(hpo3) ;
System.out.println(h) ; // 输出房子价格
h.setPrice(666666) ; // 修改房子价格
System.out.println(h) ; // 输出房子价格
}
};

运行结果:
房子价格为:1000000.0
购房者C观察到价格更改为:666666.0
购房者B观察到价格更改为:666666.0
购房者A观察到价格更改为:666666.0
房子价格为:666666.0

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值