观察者模式

来源:互联网 发布:网络现金赌博信誉平台 编辑:程序博客网 时间:2024/06/04 00:44

观察者模式——在对象之间定义一对多的依赖,这样一来,当一个对象改变状态,依赖它的对象都会收到通知,并自动更新


以下是转载于菜鸟教程

观察者模式

当对象间存在一对多关系时,则使用观察者模式(Observer Pattern)。比如,当一个对象被修改时,则会自动通知它的依赖对象。观察者模式属于行为型模式。

介绍

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

主要解决:一个对象状态改变给其他对象通知的问题,而且要考虑到易用和低耦合,保证高度的协作。

何时使用:一个对象(目标对象)的状态发生改变,所有的依赖对象(观察者对象)都将得到通知,进行广播通知。

如何解决:使用面向对象技术,可以将这种依赖关系弱化。

关键代码:在抽象类里有一个 ArrayList 存放观察者们。

应用实例: 1、拍卖的时候,拍卖师观察最高标价,然后通知给其他竞价者竞价。 2、西游记里面悟空请求菩萨降服红孩儿,菩萨洒了一地水招来一个老乌龟,这个乌龟就是观察者,他观察菩萨洒水这个动作。

优点: 1、观察者和被观察者是抽象耦合的。 2、建立一套触发机制。

缺点: 1、如果一个被观察者对象有很多的直接和间接的观察者的话,将所有的观察者都通知到会花费很多时间。 2、如果在观察者和观察目标之间有循环依赖的话,观察目标会触发它们之间进行循环调用,可能导致系统崩溃。 3、观察者模式没有相应的机制让观察者知道所观察的目标对象是怎么发生变化的,而仅仅只是知道观察目标发生了变化。

使用场景: 1、有多个子类共有的方法,且逻辑相同。 2、重要的、复杂的方法,可以考虑作为模板方法。

注意事项: 1、JAVA 中已经有了对观察者模式的支持类。 2、避免循环引用。 3、如果顺序执行,某一观察者错误会导致系统卡壳,一般采用异步方式。

实现

观察者模式使用三个类 Subject、Observer 和 Client。Subject 对象带有绑定观察者到 Client 对象和从 Client 对象解绑观察者的方法。我们创建 Subject 类、Observer 抽象类和扩展了抽象类 Observer 的实体类。

ObserverPatternDemo,我们的演示类使用 Subject 和实体类对象来演示观察者模式。

观察者模式的 UML 图

步骤 1

创建 Subject 类。

Subject.java

import java.util.ArrayList;import java.util.List;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();      }   } }

步骤 2

创建 Observer 类。

Observer.java

public abstract class Observer {   protected Subject subject;   public abstract void update();}

步骤 3

创建实体观察者类。

BinaryObserver.java

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() ) );    }}

OctalObserver.java

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() ) );    }}

HexaObserver.java

public class HexaObserver extends Observer{   public HexaObserver(Subject subject){      this.subject = subject;      this.subject.attach(this);   }   @Override   public void update() {      System.out.println( "Hex String: "       + Integer.toHexString( subject.getState() ).toUpperCase() );    }}

步骤 4

使用 Subject 和实体观察者对象。

ObserverPatternDemo.java

public class ObserverPatternDemo {   public static void main(String[] args) {      Subject subject = new Subject();      new HexaObserver(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);   }}

步骤 5

验证输出。

First state change: 15Hex String: FOctal String: 17Binary String: 1111Second state change: 10Hex String: AOctal String: 12Binary String: 1010


自己实际操作代码

package pattern;import java.util.ArrayList;import java.util.List;//测试public class Observer {public static void main(String[] args) {Weixin wTest = new Weixin();UserA aa = new UserA();UserB bb = new UserB();UserC cc = new UserC();wTest.register(aa);wTest.register(bb);wTest.setContent("1");System.out.println("-----------");wTest.remove(aa);wTest.setContent("1");System.out.println("-----------");wTest.register(cc);wTest.setContent("1");}}//步骤 1//创建 Subject 类。//Subject.javaclass Weixin{private List<User> observes = new ArrayList<>();private String content;public void setContent(String content){this.content = content;notifyAllObserves();}private void notifyAllObserves() {for (User t : observes){t.update();}}public String getContent(){return content;}public void register(User user){observes.add(user);}public void remove(User user){int n = observes.indexOf(user);for (int i=0; i<=n; i++){observes.remove(i);}}}//步骤 2//创建 Observer 类。//Observer.java//我觉得用接口比较好, 而且书上是用接口的interface User{void update();}//步骤 3//创建实体观察者类。//BinaryObserver.javaclass UserA implements User{public void update() {System.out.println("AAA");}}class UserB implements User{public void update() {System.out.println("BBB");}}class UserC implements User{public void update() {System.out.println("CCC");}}


原创粉丝点击