观察者模式

定义

定义对象间的一种一对多的依赖关系,当一个对象的状态发生改变时,所有依赖于它的对象都得到通知并被自动更新。

例:我们以新闻机构接收到新闻而后通知各个频道为例。那么新闻机构就是被观察对象,各个频道就是观察者。

实现方式

1、一般模式

① 创建Subject类
public class Subject {
   
   private List<Observer> observers 
      = new ArrayList<Observer>();
   private int state;
 
   public int getState() {
      return state;
   }
 
   public void setState(int state) {
      this.state = state;
      notifyAllObservers();
   }
 
   public void attach(Observer observer){
      observers.add(observer);      
   }
 
   public void notifyAllObservers(){
      for (Observer observer : observers) {
         observer.update();
      }
   }  
}
② 创建Observer类
public abstract class Observer {
   protected Subject subject;
   public abstract void update();
}
③ 创建实体观察者类
public class BinaryObserver extends Observer{
 
   public BinaryObserver(Subject subject){
      this.subject = subject;
      this.subject.attach(this);
   }
 
   @Override
   public void update() {
      System.out.println( "Binary String: " 
      + Integer.toBinaryString( subject.getState() ) ); 
   }
}
public class OctalObserver extends Observer{
 
   public OctalObserver(Subject subject){
      this.subject = subject;
      this.subject.attach(this);
   }
 
   @Override
   public void update() {
     System.out.println( "Octal String: " 
     + Integer.toOctalString( subject.getState() ) ); 
   }
}
④ 主方法
public class ObserverPatternDemo {
   public static void main(String[] args) {
      Subject subject = new Subject();
 
      new OctalObserver(subject);
      new BinaryObserver(subject);
 
      System.out.println("First state change: 15");   
      subject.setState(15);
      System.out.println("Second state change: 10");  
      subject.setState(10);
   }
}
⑤ 运行结果
First state change: 15
Octal String: 17
Binary String: 1111
Second state change: 10
Octal String: 12
Binary String: 1010

这个例子是参考菜鸟教程,总结得很详细。

2、实现PropertyChangeListener接口

① 创建被观察者
public class Subject {
	 private int state;
	 private String message="0";
	 private PropertyChangeSupport support;
	 
	 public Subject() {
		// TODO Auto-generated constructor stub
		 support=new PropertyChangeSupport(this);
	}
	 public int getState() {
	      return state;
	   }
	 
	   public void setState(int state) {
	      this.state = state;  
	     
	      notifyTo(String.valueOf(state));
	   }
	   
	   public void addPropertyChangeListener(PropertyChangeListener pcl) {
		   support.addPropertyChangeListener(pcl);
	   }
	   
	   public void removePropertyChangeListener(PropertyChangeListener pcl) {
		   support.removePropertyChangeListener(pcl);
	   }
	   
	   public void notifyTo(String value) {
		   support.firePropertyChange("",message,value);
		   
		   
	   }
	 
	 
}
② 创建观察者
public class BinaryObserver implements PropertyChangeListener {
	
	@Override
	public void propertyChange(PropertyChangeEvent evt) {
		// TODO Auto-generated method stub
		 System.out.println( "值更新为 "+evt.getNewValue()); 
		
	}
}
③ 主方法
public class MyObserver {
	public static void main(String[] args) {
		Subject subject=new Subject();
		BinaryObserver binaryObserver=new BinaryObserver();
		
		subject.addPropertyChangeListener(binaryObserver);
		subject.setState(10);
		subject.setState(15);

	}

}
④ 运行结果
值更新为 10
值更新为 15

这个观察者类就写了一个,懒得写第二个了。

观察者模式我们用图表达出来就是这个样子的

很形象吧!

应用场景

  • 当一个抽象模型有两个方面,其中一个方面依赖于另一方面。将这二者封装在独立的对象中以使它们可以各自独立地改变和复用。
  • 当对一个对象的改变需要同时改变其他对象,而不知道具体有多少对象需要被改变。
  • 当一个对象必须通知其他对象,而它又不能假定其他对象是谁。换言之,不希望这些对象是紧密耦合的。

就像报社和订报纸的人
发Email给用户

优缺点

优点
观察者模式解除了主题和具体观察者的耦合,让耦合的双方都依赖于抽象,而不是依赖具体。

缺点
在应用观察者模式时需要考虑一下开发小路问题,程序中包括一个被观察者和多个被观察者,开发和调试比较复杂,而且Java中的消息的通知默认是顺序执行的,一个观察者的卡顿会影响整体的执行效率。在这种情况下,一般考虑采用异步的方式。 [2]

posted @ 2020-02-09 22:39  我的小鱼干嘞  阅读(146)  评论(0编辑  收藏  举报