Java设计模式--观察者模式

观察者模式(别名:依赖,发布-订阅)

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

Observer Pattern(Another Name: Dependents, Publish-Subscribe)

Define a one-to-many dependency between objects so that when one object changes state, all its dependents are notified and updated automatically.

类图

模式的结构与使用

观察者模式的结构中包括四个角色。
+ 主题(Subject):主题是一个接口,该接口规定了具体主题需要实现的方法,比如,添加、删除观察者以及通知观察者更新数据的方法。
+ 具体主题(Concrete Subject):具体主题是实现主题接口类的一个实例,该实例包含有可以经常发生变化的数据。具体主题需使用一个集合,比如ArrayList,存放观察者的引用,以便数据变化时通知具体观察者。
+ 观察者(Observer):观察者一个接口或者抽象类,该接口规定了具体观察者用来更新数据的方法。
+ 具体观察者(Concrete Observer):具体观察者是实现观察者接口类的一个实例。具体观察者包含有可以存放具体主题引用的主题接口变量,以便具体观察者让具体主题将自己的引用添加到具体主题中,使自己成为它的观察者,或让这个具体主题将自己从具体主题的集合中删除,使自己不再是它的观察者。

简单的例子(推数据)

Subject接口类Subject.java

package Observer;

public interface Subject {
    public void addObserver(Observer o);
    public void deleteObserver(Observer o);
    public void notifyObservers();
}

ConcreteSubject的实现类SeekJobCenter.java

package Observer;

import java.util.ArrayList;

public class SeekJobCenter implements Subject {
    String mess;
    ArrayList<Observer> personList;
    boolean changed;

    public SeekJobCenter() {
        personList = new ArrayList<Observer>();
        mess = "";
        changed = false;
    }

    @Override
    public void addObserver(Observer o) {
        if (!personList.contains(o)) {
            personList.add(o);
        }
    }

    @Override
    public void deleteObserver(Observer o) {
        if (personList.contains(o)) {
            personList.remove(o);
        }
    }

    @Override
    public void notifyObservers() {
        if (changed) {
            for (int i = 0; i < personList.size(); i++) {
                Observer o = personList.get(i);
                o.hearTelephone(this.mess);
            }
            changed = false;
        }
    }

    public void giveNewMessage(String str) {
        if (str.equals(mess)) {
            System.out.println("重复信息");
            changed = false;
        } else {
            this.mess = str;
            changed = true;
        }
    }
}

Observer的接口类Observer.java

package Observer;

public interface Observer {
    public void hearTelephone(String mess);
}

ConcreteObserver的实现类UniversityStudent.java

package Observer;

public class UniversityStudent implements Observer {

    Subject subject;

    public UniversityStudent(Subject subject) {
        this.subject = subject;
        subject.addObserver(this);
    }

    @Override
    public void hearTelephone(String mess) {
        System.out.println("我是一个大学生");
        System.out.println(mess);
    }

}

ConcreteObserver的实现类MiddleSchoolStudent.java

package Observer;

public class MiddleSchoolStudent implements Observer {

    Subject subject;

    public MiddleSchoolStudent(Subject subject) {
        this.subject = subject;
        subject.addObserver(this);
    }

    @Override
    public void hearTelephone(String mess) {
        System.out.println("我是一名中学生");
        if (mess.contains("中学生")) {
            System.out.println(mess);
        } else {
            System.out.println("我是中学生,这次信息中没有我需要的信息");
        }
    }
}

测试类Application.java

package Observer;

public class Application {

    public static void main(String[] args) {
        SeekJobCenter center = new SeekJobCenter();
        new UniversityStudent(center);
        new MiddleSchoolStudent(center);
        center.giveNewMessage("大学生工作");
        center.notifyObservers();
        center.giveNewMessage("大学生工作");
        center.notifyObservers();
        center.giveNewMessage("中学生工作");
        center.notifyObservers();
    }
}

观察者模式中的“推”数据与“拉”数据

1.推数据方式

推数据方式是指具体主题将变化后的数据全部交给具体观察者,即将变化后的数据传给具体观察者用于更新数据方法的参数。当具体主题认为具体观察者需要这些变化后的全部数据时往往采用推数据方式。

2.拉数据方式

拉数据方式是指具体主题不将变化后的数据交给具体观察者,而是提供了获得这些数据的方法,具体观察者在得到通知后,可以调用具体主题提供的方法得到数据(观察者自己数据“拉”过来),但需要自己判断数据是否发生变化。当具体主题不知道具体观察者是需要这些变换后的数据时往往采用拉数据的方式。

简单的例子(拉数据)

Subject接口类Subject.java

package ObserverPull;

public interface Subject {
    public void addObserver(Observer o);
    public void deleteObserver(Observer o);
    public void notifyObserver();
}

ConcreteSubject的实现类ShopSubject.java

package ObserverPull;

import java.util.ArrayList;

public class ShopSubject implements Subject {

    String goodsName;
    double oldPrice, newPrice;

    public String getGoodsName() {
        return goodsName;
    }

    public double getOldPrice() {
        return oldPrice;
    }

    public double getNewPrice() {
        return newPrice;
    }

    public ArrayList<Observer> getCustomerList() {
        return customerList;
    }

    ArrayList<Observer> customerList;

    public ShopSubject() {
        customerList = new ArrayList<Observer>();
    }

    @Override
    public void addObserver(Observer o) {
        if (!customerList.contains(o)) {
            customerList.add(o);
        }
    }

    @Override
    public void deleteObserver(Observer o) {
        if (customerList.contains(o)) {
            customerList.remove(o);
        }
    }

    @Override
    public void notifyObserver() {
        for (int i = 0; i < customerList.size(); i++) {
            customerList.get(i).update();
        }
    }

    public void setDiscountGoods(String name, double oldP, double newP) {
        goodsName = name;
        oldPrice = oldP;
        newPrice = newP;
        notifyObserver();
    }
}

Observer的接口类Observer.java

package ObserverPull;

public interface Observer {
    public abstract void update();
}

ConcreteObserver的实现类CustomerOne.java

package ObserverPull;

public class CustomerOne implements Observer {

    Subject subject;
    String goodsName,personName;

    public CustomerOne(Subject subject, String personName) {
        this.subject = subject;
        this.personName = personName;
        subject.addObserver(this);
    }

    @Override
    public void update() {
        if (subject instanceof ShopSubject) {
            goodsName = ((ShopSubject) subject).getGoodsName();
            System.out.println(personName + "只对打折商品的名字感兴趣");
            System.out.println("打折商品是:" + goodsName);
        }
    }
}

ConcreteObserver的实现类CustomerTwo.java

package ObserverPull;

public class CustomerTwo implements Observer {
    Subject subject;
    String personName;
    double oldP,newP;

    public CustomerTwo(Subject subject, String personName) {
        this.subject = subject;
        this.personName = personName;
        subject.addObserver(this);
    }

    @Override
    public void update() {
        if (subject instanceof ShopSubject) {
            oldP = ((ShopSubject) subject).getOldPrice();
            newP = ((ShopSubject) subject).getNewPrice();
            System.out.println(personName + "只对商品的新旧价格感兴趣");
            System.out.println("商品旧价格是:" + oldP);
            System.out.println("商品新价格是:" + newP);
        }
    }
}

测试类Application.java

package ObserverPull;

public class Application {

    public static void main(String[] args) {
        ShopSubject shop = new ShopSubject();
        new CustomerOne(shop, "yxx");
        new CustomerTwo(shop, "yn");
        shop.setDiscountGoods("pen", 20, 10);
        shop.setDiscountGoods("相机", 1000, 800);
    }
}

观察者与多主体

一个具体观察者可以依赖于多个主题,当所有依赖的任何具体主题的数据发生变化时,该观察者都能得到通知。多主题所涉及的主要问题是观察者如何处理主题中变化后的数据,因为,不同的具体主题所包含有的数据结构可能很大不同。
在处理多主题时,主题应当采用拉数据方式,观察者接口可以将更新数据方法的参数类型设置为主题接口类型,比如update(Subject subject),即具体主题数据放生变化时将自己的引用传递给具体的观察者,然后具体观察者让这个具体主题调用有关的方法返回该具体主题中的数据。

观察者模式的优点

  • 具体主题和具体观察者是松耦合关系。由于主题接口仅仅依赖于观察者接口,因此具体主题只是知道它的观察者是实现观察者接口的某个类的实例,但不需要知道具体哪个类。同样,由于观察者仅仅依赖于主题接口,因此具体观察者只是知道它依赖的主题是实现主题接口的某个类的实例,但不需要知道具体是哪一个类。
  • 观察模式满足“开-闭原则”。主题接口仅仅依赖于观察者接口,这样,就可以让创建具体主题的类也仅仅是依赖于观察者接口,因此如果增加新的实现观察者接口的类,不必修改创建具体主题的类的代码。同样,创建具体观察者的类仅仅依赖于主题接口,如果增加新的实现主题接口的类,也不必修改创建具体观察者类的代码。

适用观察者模式的情景

  • 当一个对象的数据更新时需要通过其他对象,但这个对象又不希望和被通知的那些对象形成紧耦合。
  • 当一个对象的数据更新时,这个对象需要让其他对象也各自更新自己的数据,但这个对象不知道具体有多少对象需要更新数据。

下载源码请到

MyGitHub

时间: 2024-08-01 12:29:32

Java设计模式--观察者模式的相关文章

Java设计模式----观察者模式详解

[正文] 一.观察者模式的定义: 简单地说,观察者模式定义了一个一对多的依赖关系,让一个或多个观察者对象监听一个主题对象.这样一来,当被观察者状态发生改变时,需要通知相应的观察者,使这些观察者对象能够自动更新.例如:GUI中的事件处理机制采用的就是观察者模式.   二.观察者模式的实现: Subject(被观察的对象接口):规定ConcreteSubject的统一接口 ; 每个Subject可以有多个Observer: ConcreteSubject(具体被观察对象):维护对所有具体观察者的引用

Java设计模式开发中使用观察者模式的实例教程_java

观察者模式是软件设计模式中的一种,使用也比较普遍,尤其是在GUI编程中.关于设计模式的文章,网络上写的都比较多,而且很多文章写的也不错,虽然说有一种重复早轮子的嫌疑,但此轮子非彼轮子,侧重点不同,思路也不同,讲述方式也不近相同.关键要素 主题: 主题是观察者观察的对象,一个主题必须具备下面三个特征. 持有监听的观察者的引用 支持增加和删除观察者 主题状态改变,通知观察者 观察者: 当主题发生变化,收到通知进行具体的处理是观察者必须具备的特征. 为什么要用这种模式 这里举一个例子来说明,牛奶送奶站

实例解析观察者模式及其在Java设计模式开发中的运用_java

一.观察者模式(Observer)的定义: 观察者模式又称为订阅-发布模式,在此模式中,一个目标对象管理所有相依于它的观察者对象,并且在它本身的状态改变时主动发出通知.这通常透过呼叫各观察者所提供的方法来实现.此种模式通常被用来事件处理系统. 1.观察者模式的一般结构 首先看下观察者模式的类图描述: 观察者模式的角色如下: Subject(抽象主题接口):定义了主题类中对观察者列表的一系列操作, 包括增加,删除, 通知等. Concrete Subject(具体主题类): Observer(抽象

深入解析Java设计模式编程中观察者模式的运用_java

定义:定义对象间一种一对多的依赖关系,使得当每一个对象改变状态,则所有依赖于它的对象都会得到通知并自动更新.类型:行为类模式类图: 在软件系统中经常会有这样的需求:如果一个对象的状态发生改变,某些与它相关的对象也要随之做出相应的变化.比如,我们要设计一个右键菜单的功能,只要在软件的有效区域内点击鼠标右键,就会弹出一个菜单:再比如,我们要设计一个自动部署的功能,就像eclipse开发时,只要修改了文件,eclipse就会自动将修改的文件部署到服务器中.这两个功能有一个相似的地方,那就是一个对象要时

用Java设计模式中的观察者模式开发微信公众号的例子_java

还记得警匪片上,匪徒们是怎么配合实施犯罪的吗?一个团伙在进行盗窃的时候,总有一两个人在门口把风--如果有什么风吹草动,则会立即通知里面的同伙紧急撤退.也许放风的人并不一定认识里面的每一个同伙:而在里面也许有新来的小弟不认识这个放风的.但是这没什么,这个影响不了他们之间的通讯,因为他们之间有早已商定好的暗号. 呵呵,上面提到的放风者.偷窃者之间的关系就是观察者模式在现实中的活生生的例子. 观察者(Observer)模式又名发布-订阅(Publish/Subscribe)模式.GOF给观察者模式如下

学习Java设计模式之观察者模式_java

观察者模式:对象间的一种一对多的依赖关系,让多个观察者对象同时监听某一个主题对象(被观察). 以便一个对象的状态发生变化时,所有依赖于它的对象都得到通知并发生相应的变化. 观察者模式有很多实现方式:该模式必须包含观察者和被观察对象两种角色.观察者和被观察者之间存在"观察"的逻辑关系,当被观察者发生改变的时候,观察者就会观察到这样的变化,发出相应的改变. /** * 观察者接口:观察者,需要用到观察者模式的类需实现此接口 */ public interface Observer { pu

java设计模式之观察者模式_java

        观察者模式又称发布-订阅(Publish/Subscribe)模式,定义了一种一对多的依赖关系,让多个观察者对象同时监听某一个主题对象.这个主题对象在状态发生变化时,会通知所有观察者对象,使他们能够自动更新自己.将一个系统分割成一系列相互协作的类有一个很不好的副作用,那就是需要维护相关对象间的一致性.我们不希望为了维持一致性而使各类紧密耦合,这样会给维护.扩展和复用都带来不便.观察者模式所做的工作其实就是在解除耦合,让耦合的双方都依赖于抽象,而不是依赖于具体. 观察者模式是实际中

Java设计模式(十五)----观察者模式

观察者模式 一.定义 二.结构 具体案例 推模型和拉模型 三.Java提供的对观察者模式的支持 Observer接口 Observable类 一.定义 观察者模式是对象的行为模式,又叫发布-订阅(Publish/Subscribe)模式.模型-视图(Model/View)模式.源-监听器(Source/Listener)模式或从属者(Dependents)模式. 观察者模式定义了一种一对多的依赖关系,让多个观察者对象同时监听某一个主题对象.这个主题对象在状态上发生变化时,会通知所有观察者对象,使

您的设计模式,我们的设计模式 java设计模式

http://download.csdn.net/download/yangxin00000000/3212729   您的设计模式,我们的设计模式 java设计模式