观察者模式(Observer)

xiaohai 2021-06-08 11:20:34 2722人围观 标签: 设计模式 
简介指多个对象间存在一对多的依赖关系,当一个对象的状态发生改变时,所有依赖于它的对象都得到通知并被自动更新。这种模式有时又称作发布-订阅模式、模型-视图模式,它是对象行为型模式。

模式定义

指多个对象间存在一对多的依赖关系,当一个对象的状态发生改变时,所有依赖于它的对象都得到通知并被自动更新。这种模式有时又称作发布-订阅模式、模型-视图模式,它是对象行为型模式。

模式优缺点

  • 优点

      降低了目标与观察者之间的耦合关系,两者之间是抽象耦合关系
      目标与观察者之间建立了一套触发机制
      符合开闭原则
    
  • 缺点

      目标与观察者之间的依赖关系并没有完全解除,而且有可能出现循环引用
      当观察者对象很多时,通知的发布会花费很多时间,影响程序的效率
    

应用场景

  • 对象间存在一对多关系,一个对象的状态发生改变会影响其他对象
  • 当一个抽象模型有两个方面,其中一个方面依赖于另一方面时,可将这二者封装在独立的对象中以使它们可以各自独立地改变和复用

模式结构

  • 抽象主题(Subject)角色:也叫抽象目标类,它提供了一个用于保存观察者对象的聚集类和增加、删除观察者对象的方法,以及通知所有观察者的抽象方法。
  • 具体主题(Concrete Subject)角色:也叫具体目标类,它实现抽象目标中的通知方法,当具体主题的内部状态发生改变时,通知所有注册过的观察者对象。
  • 抽象观察者(Observer)角色:它是一个抽象类或接口,它包含了一个更新自己的抽象方法,当接到具体主题的更改通知时被调用。
  • 具体观察者(Concrete Observer)角色:实现抽象观察者中定义的抽象方法,以便在得到目标的更改通知时更新自身的状态。

模式实现

package com.company.observer;

import javax.security.auth.Subject;
import java.util.ArrayList;
import java.util.List;

public class ObserverPattern {
    public static void main(String[] args) {
        Object subject = new ConcreteObject();

        //实例化两个观察者
        ConcreteObserver1 observer1 = new ConcreteObserver1();
        ConcreteObserver2 observer2 = new ConcreteObserver2();

        subject.addObserver(observer1);
        subject.addObserver(observer2);

        subject.notifyObserver();

        subject.removeObserver(observer2);
        subject.notifyObserver();
    }
}

//抽象目标
abstract class Object{
    protected List<Observer> observers = new ArrayList<>();

    //添加观察者
    public void addObserver(Observer observer){
        observers.add(observer);
    }

    //删除观察者
    public void removeObserver(Observer observer){
        observers.remove(observer);
    }

    public abstract void notifyObserver();
}

//具体目标
class ConcreteObject extends Object{

    @Override
    public void notifyObserver() {
        for(Observer observer:observers){
            observer.update();
        }
    }
}


//观察者接口
interface Observer{
    void update();//操作
}

//具体观察者1
class ConcreteObserver1  implements Observer{

    @Override
    public void update() {
        System.out.println("具体观察者1作出反应!");
    }
}

//具体观察者2
class ConcreteObserver2  implements Observer{
    @Override
    public void update() {
        System.out.println("具体观察者2作出反应!");
    }
}

运行结果:

具体观察者1作出反应!
具体观察者2作出反应!
具体观察者1作出反应!