观察者(Observer)模式又名发布-订阅(Publish/Subscribe)模式。GOF给观察者模式如下定义:定义对象间的一种一对多的依赖关系,当一个对象的状态发生改变时,所有依赖于它的对象都得到通知并被自动更新。
在这里先讲一下面向对象设计的一个重要原则——单一职责原则。因此系统的每个对象应该将重点放在问题域中的离散抽象上。因此理想的情况下,一个对象只做一件事情。这样在开发中也就带来了诸多的好处:提供了重用性和维护性,也是进行重构的良好的基础。
因此几乎所有的设计模式都是基于这个基本的设计原则来的。观察者模式的起源我觉得应该是在GUI和业务数据的处理上,因为现在绝大多数讲解观察者模式的例子都是这一题材。
以手机号码为例,老师的手机号码存在学生的手机里,若老师的手机号改变,她会发一条短信通知每个学生自己手机号变了。代码如下:
Subject.java
package observer;
public interface Subject {
// 注册观察者
void RegisterObserver(Observer o);
// 销毁观察者
void RemoveObserver(Observer o);
// 通知观察者
public void NoticeObserver();
}
Observer.java
package observer;
public interface Observer {
public void Update();
}
Teacher.java
package observer;
import java.util.Vector;
public class Teacher implements Subject {
private String phone;
private Vector students;
public Teacher() {
phone = "";
students = new Vector();
}
// 注册观察者
public void RegisterObserver(Observer o) {
students.add(o);
}
// 销毁观察者
public void RemoveObserver(Observer o) {
students.remove(o);
}
// 通知观察者
public void NoticeObserver() {
for (int i = 0; i < students.size(); i++)
((Observer) students.get(i)).Update();
}
public void setPhone(String phone) {
this.phone = phone;
NoticeObserver();
}
public String getPhone() {
return phone;
}
}
Student.java
package observer;
//学生(观察者)代码
public class Student implements Observer {
private String name;
private String phone;
private Teacher teacher;
public Student(String name, Teacher t) {
this.name = name;
teacher = t;
}
public void show() {
System.out.println("Name:" + name + "\nTeacher'sphone:" + phone);
}
public void Update() {
phone = teacher.getPhone();
}
}
客户端调用代码
package observer;
import java.util.Vector;
public class Client {
//客户端,测试代码
public static void main(String[] args) {
Vector students = new Vector();
Teacher t = new Teacher();
Student st = new Student("张三", t);
//添加学生张三
students.add(st);
//在老师的通讯录中添加学生张三
t.RegisterObserver(st);
st = new Student("李四", t);
//添加学生李四
students.add(st);
//在老师的通讯录中添加学生李四
t.RegisterObserver(st);
st = new Student("王五", t);
//添加学生王五
students.add(st);
//在老师的通讯录中添加学生王五
t.RegisterObserver(st);
//设置老师电话号码
t.setPhone("88803807");
//通知所有学生
for (int i = 0; i < students.size(); i++)
((Student) students.get(i)).show();
//老师电话号码改变
t.setPhone("88808880");
for (int i = 0; i < students.size(); i++)
((Student) students.get(i)).show();
}
}