java obervable_设计模式--观察者模式初探和java Observable模式

初步认识观察者模式

观察者模式又称为发布/订阅(Publish/Subscribe)模式,因此我们可以用报纸期刊的订阅来形象的说明:

报社方负责出版报纸.

你订阅了该报社的报纸,那么只要报社发布了新报纸,就会通知你,或发到你手上.

如果你不想再读报纸,可以取消订阅,这样,报社发布了新报纸就不会再通知你.

理解其实以上的概念,就可以理解观察者模式,观察者模式中有主题(Subject)和观察者(Observer),分别对应报社和订阅用户(你).观察者模式定义了对象之间的一对多的依赖关系,这样,当"一"的一方状态发生变化时,它所依赖的"多"的一方都会收到通知并且自动更新.如图:

7eb0fced36c8334e1e76a1d2cc7f3a2a.png

实现观察者模式

这里以师生关系为例,老师和学生是一对多的关系,老师给学生布置作业,这个动作作为主题事件,每当老师布置一道题时,就要自动通知到所有的学生把该题记下来,然后再布置下一道题...

fa35f481205e1a8265b7db595b99efb5.png

这是一个并不标准的类图,主题接口和实现类中,一般需要有addObserver(),deleteObserver(),notifyObserver();用来注册删除观察者以及在主题状态发生变化时通知所有的观察者对象.

接下来进行代码实现:

Subject接口:

packagecom.wang.observer;//主题接口

interfaceSubject {//添加观察者

voidaddObserver(Observer obj);//移除观察者

voiddeleteObserver(Observer obj);//当主题方法改变时,这个方法被调用,通知所有的观察者

voidnotifyObserver();

}

Oserver接口:

packagecom.wang.observer;interfaceObserver {//当主题状态改变时,会将一个String类型字符传入该方法的参数,每个观察者都需要实现该方法

public voidupdate(String info);

}

Subject接口实现类TeacherSubject:

packagecom.wang.observer;importjava.util.ArrayList;importjava.util.List;public class TeacherSubject implementsSubject {//用来存放和记录观察者

private List observers=new ArrayList();//记录状态的字符串

privateString info;

@Overridepublic voidaddObserver(Observer obj) {

observers.add(obj);

}

@Overridepublic voiddeleteObserver(Observer obj) {int i =observers.indexOf(obj);if(i>=0){

observers.remove(obj);

}

}

@Overridepublic voidnotifyObserver() {for(int i=0;i

Observer o=(Observer)observers.get(i);

o.update(info);

}

}//布置作业的方法,在方法最后,需要调用notifyObserver()方法,通知所有观察者更新状态

public voidsetHomework(String info){this.info=info;

System.out.println("今天的作业是"+info);this.notifyObserver();

}

}

Observer接口实现类StudentObserver:

packagecom.wang.observer;public class StudentObserver implementsObserver {

//保存一个Subject的引用,以后如果可以想取消订阅,有了这个引用会比较方便privateTeacherSubject t;//学生的姓名,用来标识不同的学生对象

privateString name;//构造器用来注册观察者

publicStudent(String name,Teacher t) {this.name=name;this.t =t;//每新建一个学生对象,默认添加到观察者的行列

t.addObserver(this);

}

@Overridepublic voidupdate(String info) {

System.out.println(name+"得到作业:"+info);

}

}

测试类TestObserver:

packagecom.wang.observer;public classTestObserver {public static voidmain(String[] args) {

TeacherSubject teacher=newTeacherSubject();

StudentObserver zhangSan=new StudentObserver("张三", teacher);

StudentObserver LiSi=new StudentObserver("李四", teacher);

StudentObserver WangWu=new StudentObserver("王五", teacher);

teacher.setHomework("第二页第六题");

teacher.setHomework("第三页第七题");

teacher.setHomework("第五页第八题");

}

}

打印结果:

今天的作业是第二页第六题

张三得到作业:第二页第六题

李四得到作业:第二页第六题

王五得到作业:第二页第六题

今天的作业是第三页第七题

张三得到作业:第三页第七题

李四得到作业:第三页第七题

王五得到作业:第三页第七题

今天的作业是第五页第八题

张三得到作业:第五页第八题

李四得到作业:第五页第八题

王五得到作业:第五页第八题

从打印结果看,每当老师布置作业的状态改变,就会通知每一个学生.以上就是一个简单的观察者模式的实现.

java内置的观察者模式:

在java.util包中包含有基本的Observer接口和Observable抽象类.功能上和Subject接口和Observer接口类似.不过在使用上,就方便多了,因为许多功能比如说注册,删除,通知观察者的那些功能已经内置好了.

使用javaAPI的观察者模式需要明白这么几件事情:

如何使对象变为观察者?

实现观察者接口(java.util.Observer),然后调用Observable对象的addObserver()方法.不想再当观察者时,调用deleteObserver()就可以了.

被观察者(主题)如何发出通知?

第一步:先调用setChanged()方法,标识状态已经改变的事实.

第二步:调用notifyObservers()方法或者notifyObservers(Object arg),这就牵扯到推(push)和拉(pull)的方式传送数据.如果想用push的方式"推"数据给观察者,可以把数据当做数据对象传送给notifyObservers(Object arg)方法,其中的arg可以为任意对象,意思是你可以将任意对象传送给每一个观察者.如果调用不带参数的notifyObserver()方法,则意味着你要使用pull的方式去主题对象中"拉"来所需要的数据.

观察者如何接收通知?

观察者只需要实现一个update(Observable o,Object arg)方法,第一个参数o,是指定通知是由哪个主题下达的,第二个参数arg就是上面notifyObserver(Object arg)里传入的数据,如果不传该值,arg为null.

下面使用java内置API实现上面我所写的老师和学生的例子:

被观察者TeacherSubject:

packagecom.wang.observer1;importjava.util.Observable;public class Teacher extendsObservable {//布置作业的状态信息字符串

privateString info;public voidsetHomework(String info) {this.info=info;

System.out.println("布置的作业是"+info);

setChanged();

notifyObservers();

}publicString getInfo() {returninfo;

}

}

观察者StudentObserver:

packagecom.wang.observer1;importjava.util.Observable;importjava.util.Observer;public class Student implementsObserver{privateObservable ob;privateString name;publicStudent(String name,Observable ob) {this.ob =ob;this.name=name;

ob.addObserver(this);

}

@Overridepublic voidupdate(Observable o, Object arg) {

Teacher t=(Teacher)o;

System.out.println(name+"得到作业信息:"+t.getInfo());

}

}

测试代码和打印结果我就不再写了,和上面的例子是一样一样的,在这个例子中我使用的是"pull"的方式拉数据,在需要传递状态的TeacherSubject中定义了一个info字符串的get方法,在观察者对象中调用get方法得到所需数据,如果希望使用push的方式,只需要在TeacherSubject类的notifyOservers()方法中传入String类型的info字符串即可在update()方法中直接通过第二个参数获取到arg,即使前面传过来的info字符串.

观察者模式的好处

观察者模式提供了一种对象设计,让主题和观察者之间耦合度降得很低,为什么呢?关于观察者的一切,主题只知道观察者实现了Observer接口,并不需要观察者具体的类是谁,做了什么或者其他细节.

这样的话,由于松耦合,改变主题或者观察者其中一方,并不会影响另一方,只要他们之间的接口仍被遵守,就可以自由地改变它.

降低对象之间的耦合度,也是面设对象设计的一个很重要的原则.

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值