观察者模式

Posted 温布利往事

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了观察者模式相关的知识,希望对你有一定的参考价值。

一、概念

  观察者模式,又被称为发布—订阅模式、源—收听者模式,是软件设计模式的一种。在此种模式中,一个目标物件管理所有相依与它的观察者物件,并且在它本身的状态改变时主动发出同时,此种模式通常被用来实现事件处理系统。

二、观察者模式组成

  观察者模式一般包含以下四种角色:

  抽象主题角色Watched:把所有对观察者对象的引用保存在一个集合中,每个抽象主题角色都可以有任意数量的观察者。抽象主题提供一个接口,可以增加和删除观察这角色。

  抽象观察者角色Watcher:为所有具体的观察者定义一个接口,在得到主题的通知更新自己。

  具体主题角色ConcreteWatched:在具体主题内部状态改变时,给所有登记过的观察者发出通知。

  具体观察者角色ConcreteWatcher:该角色实现抽象观察者角色所要求的更新接口,以便使自身的状态与主题的状态相协调。

三、观察者模式实现方式

  观察者模式有很多实现方式,从根本上说,该模式必须包含两个角色,观察者和被观察对象。比较直观的一种是使用“注册”—“通知”—“撤销通知”的形式。

  观察者将自己注册到被观察对象中,被观察者将观察者放在一个容器中。

  被观察者对象发生某种变化时,从容器中得到所有被注册过的观察者,将变化通知观察者。

  观察者将自己注册到被观察者的容器时,被观察者不应该过问观察者的具体类型,而是应该使用观察者的接口。这样的好处是:假定程序还有别的观察者,那么只要这个观察者也是相同的接口即可。一个被观察者可以对应多个观察者,当被观察者发生变化的时候,他可以将消息一一通知给所有的观察者。如下图所示:

  

  具体到一个应用场景就是:读者订阅新闻,每一个读者都可以通过regist来订阅新闻,通过remove来取消订阅,当有新闻到来时,使用sendNews来通知每一位订阅的读者来收取新闻。

  下面用代码来实现上面的实例:

  抽象主题角色Watched:新闻接口,定义订阅、取消订阅和推送新闻三个方法

public interface Watched
{
    //订阅新闻
    void registeSubscriber(Watcher f_subscriber);
    //取消订阅
    void removeSubscriber(Watcher f_subscriber);
    //推送新闻
    void sendNews();
}

  抽象观察者角色Watcher:读者接口,定义了获取新闻的方法

public interface Watcher
{
    //获取新闻
    void updateNews();
}

  具体主题角色ConcreteWatched,即被观察者,实现了Watched接口:

import java.util.ArrayList;
import java.util.List;

/*
 * 具体被观察者
 * 新闻发布者
 */
public class ConcreteWatched implements Watched
{
    private List<Watcher> subList=new ArrayList<Watcher>();
    @Override
    public void registeSubscriber(Watcher f_subscriber)
    {
        subList.add(f_subscriber);    
    }
    @Override
    public void removeSubscriber(Watcher f_subscriber)
    {
        subList.remove(f_subscriber);        
    }
    @Override
    public void sendNews()
    {
     System.out.println("开始本轮新闻推送!");
for (Watcher watcher : subList) { watcher.updateNews(); } System.out.println("本次新闻推送结束!"); } }

  具体观察者角色ConcreteWatcher,即观察者对象

/*
 * 具体的观察者
 * 读者
 */
public class ConcreteWatcher implements Watcher
{
    private String username;
        
    public ConcreteWatcher(String username)
    {
        super();
        this.username = username;
    }
    @Override
    public void updateNews()
    {
        System.out.println(username+"获取到最新的新闻了!");
        
    }
}

  下面是一个测试用例:

import org.junit.Test;

public class WatchTest
{
    @Test
    public void test()
    {
        ConcreteWatched concreteWatched=new ConcreteWatched();
        Watcher watcher1=new ConcreteWatcher("读者A");
        Watcher watcher2=new ConcreteWatcher("读者B");
        Watcher watcher3=new ConcreteWatcher("读者C");
        
        //订阅新闻
        concreteWatched.registeSubscriber(watcher1);
        concreteWatched.registeSubscriber(watcher2);
        concreteWatched.registeSubscriber(watcher3);
        
        //推送新闻
        concreteWatched.sendNews();
        //读者A取消订阅
        concreteWatched.removeSubscriber(watcher1);

        //再次推送新闻
        concreteWatched.sendNews();
    }
}

  执行结果:

  

四、参考资料

  1、http://www.cnblogs.com/mengdd/archive/2013/02/07/2908929.html

以上是关于观察者模式的主要内容,如果未能解决你的问题,请参考以下文章

未调用 LiveData 观察者

Java设计模式补充:回调模式事件监听器模式观察者模式(转)

如何为片段设置观察者

永远观察实时数据的片段

设计模式观察者模式 ( 简介 | 适用场景 | 优缺点 | 代码示例 )

观察者模式