行为型:观察者模式及相关应用


观察者模式(Observer)

定义了对象之间的一对多依赖,让多个观察者对象同时监听某一个主题对象,当主题对象发生变化时,它的所有依赖者(观察者)都会收到通知并更新。

优缺点

  • 优点:观察者和被观察者之间建立一个抽象的耦合;支持广播通信。
  • 缺点:观察者之间有过多的细节依赖、提高时间消耗及程序复杂度。

应用场景

每个课程有一名老师,而课程的学生可能提出许多问题,因此创建三个类CourseQuestionTeacher。其中课程应该作为被观察者,而老师由于要回答学生们的问题,因此作为观察者时刻观察着。

课程类Course

public class Course extends Observable {
    private String courseName;

    public Course(String courseName) {
        this.courseName = courseName;
    }

    public String getCourseName() {
        return courseName;
    }

    public void setCourseName(String courseName) {
        this.courseName = courseName;
    }

    public void produceQuestion(Course course, Question question){
        System.out.println(question.getUserName() + "在" + course.getCourseName() + "提交了一个问题");
        setChanged();
        notifyObservers(question);
    }
}

问题类Question

public class Question {
    private String userName;
    private String questionContent;

    public String getUserName() {
        return userName;
    }

    public void setUserName(String userName) {
        this.userName = userName;
    }

    public String getQuestionContent() {
        return questionContent;
    }

    public void setQuestionContent(String questionContent) {
        this.questionContent = questionContent;
    }
}

教师类Teacher

public class Teacher implements Observer {
    private String teacherName;

    public Teacher(String teacherName) {
        this.teacherName = teacherName;
    }

    public void update(Observable o, Object arg) {
        Course course = (Course)o;
        Question question = (Question) arg;
        System.out.println(teacherName + "老师的" + course.getCourseName() + "课程接收到一个" + question.getUserName() + "提交的问答:" + question.getQuestionContent());
    }
}

可以看出,以上代码通过继承Observable类和实现Observer接口实现了观察者模式。

Observable中有两个方法对Observer特别重要,一个是setChanged()方法用来设置一个内部标志位表示数据发生了变化,一个是notifyObservers()方法会去调用一个列表中所有的Observerupdate()方法,通知它们数据发生了变化。

客户端类Test

public class Test {
    public static void main(String[] args) {
        Course course = new Course("Java设计模式课程");
        Teacher teacher1 = new Teacher("Alpha");
        Teacher teacher2 = new Teacher("Belta");

        course.addObserver(teacher1);
        course.addObserver(teacher2);
        Question question = new Question();
        question.setUserName("cenjie");
        question.setQuestionContent("Java的主函数如何编写");

        course.produceQuestion(course, question);
    }
}

Observable通过addObserver()方法把任意多个Observer添加到这个列表中。

运行结果:

cenjie在Java设计模式课程提交了一个问题
Belta老师的Java设计模式课程课程接收到一个cenjie提交的问答:Java的主函数如何编写
Alpha老师的Java设计模式课程课程接收到一个cenjie提交的问答:Java的主函数如何编写

可以看出,当被观察者course对象发生变化时,teacher1teacher2这两个观察者都得到了通知。

参考资料

  • 弗里曼. Head First 设计模式 [M]. 中国电力出版社, 2007.
  • 慕课网java设计模式精讲 Debug 方式+内存分析
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值