`

观察者设计模式

阅读更多
先来看下下报纸和杂志的订阅:
(1)报社:出版报纸和杂志
(2)订阅者:向某家报社订阅报纸和杂志,只要报社出版了新的报纸,订阅者就会收到最新的报纸和杂志。
(3)报社具有添加和删除订阅者的功能(其实应该是订阅者具有订阅和退订的功能,这个主动权应该是订阅者而不是报社,报社也应该对外开放这样的方法)
下面就让我们来简单实现上述的描述:
报社:PublishingHouse 如下:

package com.lg.design.obser;

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

public class PublishingHouse {
	
	private List<Subscriber> subscribers=new ArrayList<Subscriber>();
	
	public void add(Subscriber subscriber){
		if(subscriber!=null && !subscribers.contains(subscriber)){
			subscribers.add(subscriber);
		}
	}
	
	public void delete(Subscriber subscriber){
		if(subscriber!=null){
			subscribers.remove(subscriber);
		}
	}

	public void produceNewspaper(String newspaper) {
		notifySubscribers(newspaper);
	}

	private void notifySubscribers(String newspaper) {
		for(Subscriber subscriber:subscribers){
			subscriber.receiveNewspaper(newspaper);
		}
	}
	
}

PublishingHouse 有添加和删除订阅者Subscriber的方法,同时在生产报纸时来通知所有的订阅者。
订阅者:Subscriber如下:

package com.lg.design.obser;

public interface Subscriber {

	public void receiveNewspaper(String newspaper);
	
}

订阅者是一个接口,订阅到报纸后如何处理不同的订阅者有不同的实现。
如某类订阅者 ASubscriber:

package com.lg.design.obser;

public class ASubscriber implements Subscriber{

	@Override
	public void receiveNewspaper(String newspaper) {
		System.out.println("A receive newspaper:"+newspaper);
	}

}

再如某类订阅者 BSubscriber:
package com.lg.design.obser;

public class BSubscriber implements Subscriber{

	@Override
	public void receiveNewspaper(String newspaper) {
		System.out.println("B receive newspaper:"+newspaper);
	}

}

测试方法如下:
public static void main(String[] args){
		PublishingHouse publishingHouse=new PublishingHouse();
		Subscriber a=new ASubscriber();
		Subscriber b=new BSubscriber();
		publishingHouse.add(a);
		publishingHouse.add(b);
		
		publishingHouse.produceNewspaper("第一天的报纸");
		publishingHouse.produceNewspaper("第二天的报纸");
	}

输出的结果为:
A receive newspaper:第一天的报纸
B receive newspaper:第一天的报纸
A receive newspaper:第二天的报纸
B receive newspaper:第二天的报纸

观察者模式其实很简单,就是报社维护了一个订阅者集合,一旦有新的报纸,就会去遍历那个集合,调用集合里元素的通用方法receiveNewspaper(接口方法)。报社的produceNewspaper方法就是将上述过程进行了封装而已。

上述的例子很简单,同时有很多问题。下面来说说JDK自身已实现的观察者模式:
被观察者Observable(如报社),观察者Observer(如订阅者)
观察者Observer如下:

public interface Observer {
    
    void update(Observable o, Object arg);
}

被观察者Observable如下:
package java.util;
public class Observable {
    private boolean changed = false;
    private Vector obs;

    public Observable() {
        obs = new Vector();
    }

    public synchronized void addObserver(Observer o) {
        if (o == null)
            throw new NullPointerException();
        if (!obs.contains(o)) {
            obs.addElement(o);
        }
    }

    public synchronized void deleteObserver(Observer o) {
        obs.removeElement(o);
    }

    public void notifyObservers() {
        notifyObservers(null);
    }

    public void notifyObservers(Object arg) {
        /*
         * a temporary array buffer, used as a snapshot of the state of
         * current Observers.
         */
        Object[] arrLocal;

        synchronized (this) {
            /* We don't want the Observer doing callbacks into
             * arbitrary code while holding its own Monitor.
             * The code where we extract each Observable from
             * the Vector and store the state of the Observer
             * needs synchronization, but notifying observers
             * does not (should not).  The worst result of any
             * potential race-condition here is that:
             * 1) a newly-added Observer will miss a
             *   notification in progress
             * 2) a recently unregistered Observer will be
             *   wrongly notified when it doesn't care
             */
            if (!changed)
                return;
            arrLocal = obs.toArray();
            clearChanged();
        }

        for (int i = arrLocal.length-1; i>=0; i--)
            ((Observer)arrLocal[i]).update(this, arg);
    }
    public synchronized void deleteObservers() {
        obs.removeAllElements();
    }

    protected synchronized void setChanged() {
        changed = true;
    }

    protected synchronized void clearChanged() {
        changed = false;
    }

    public synchronized boolean hasChanged() {
        return changed;
    }

    public synchronized int countObservers() {
        return obs.size();
    }
}

它使用一个Vector集合来存放所有的Observer,具有添加删除Observer的功能。同时还有一个changed属性,用来标示状态是否发生变化,所以在执行notifyObservers通知前要把该状态改为true,为了保证线程安全,setChanged、hasChanged、clearChanged都加上了synchronized 进行同步。再来详细看下notifyObservers过程:
arrLocal作为当时Observers的一个快照,在该方法中对状态的判断和改变进行了同步,然后按照倒序进行了事件的通知,即调用集合中每个Observer。

先说下我对jdk实现的观察者模式的看法:
(1)加入changed属性,则需要在每次想要执行通知前,提前设置setChanged即changed=true,然后执行notifyObservers才有效,这使得我们在使用时必须使setChanged方法和notifyObservers方法进行synchronized操作,使之成为“原子操作”,不然在多线程环境中,会通知失败。如线程1执行setChanged,线程2执行setChanged,线程1执行notifyObservers,它就会把changed=false,此时线程2再执行notifyObservers,就会因为changed=false而直接退出,不会发出通知,changed属性的确给我们带来了同步的麻烦,也体现在了notifyObservers方法中,如下:

public void notifyObservers(Object arg) {
        /*
         * a temporary array buffer, used as a snapshot of the state of
         * current Observers.
         */
        Object[] arrLocal;

        synchronized (this) {
            if (!changed)
                return;
            arrLocal = obs.toArray();
            clearChanged();
        }

        for (int i = arrLocal.length-1; i>=0; i--)
            ((Observer)arrLocal[i]).update(this, arg);
    }

有了changed属性,必须要在该方法中对changed的判断进行同步,否则会造成如下现象:线程1设置了setChanged方法,在执行notifyObservers时还未执行到clearChanged方法,此时线程2不用调用setChanged方法,直接调用notifyObservers,也会实现通知。
(2)Observer接口的方法,void update(Observable o, Object arg);此时传递的是Observable对象,如MyObservable继承了Observable 对象,此时我们要获取MyObservable的数据就必须要对void update(Observable o, Object arg)传进来的Observable o进行强制类型转换,而我们应该可以利用泛型来更好的设计这一个接口,避免强制类型转换。如下:

public interface MyObserver<T extends Observable>{

	public void update(T t,Object args);
}

(3)对于Observable的notifyObservers通知顺序,默认写死为倒序通知,不应该是这样的,而是,应该对开开放成接口,同时提供多种实现,如正序通知的实现和倒序通知的实现。如果还满足不了需求,我们就可以自定义顺序实现该接口,设置进Observable中。
(4)对于Observable内部使用Vector,也是不可扩展的一个地方,如果我们想换种集合就没法实现,并不是什么需求都会去使用Vector。
(5)观察者自己应该具有取消关注的权利,即微信中的订阅一样,自己有权利去关注和取消关注,所以有时观察者接口由希望有一个取消关注的方法,然而该方法的实现又很明白,不需要观察者自己来实现。这就使用到了java8中接口中默认方法的实现。如下所示:

public interface Observer {
    
    void update(Observable o, Object arg);

    default void unregister(Observable o){
         o.deleteObserver(this);
    }

    default void register(Observable o){
         o.addObserver(this);
    }
}

不知道这样写对不对,
所以对于Jdk自身实现的观察者模式扩展性并不好,同时大量使用synchronized来解决多线程问题,没有更好的使用多线程包中的同步技术。观察者模式本身很简单,所以有时候还是需要我们自己来写一个观察者模式。

以上纯属个人拙见,欢迎激烈讨论,共同进步。

若想转载请注明出处:   http://lgbolgger.iteye.com/blog/2115108
作者:iteye的乒乓狂魔
分享到:
评论

相关推荐

    ObserverDemo观察者设计模式源码

    观察者设计模式(Observer Pattern)是软件设计模式中的一种行为模式,它定义了对象之间的一对多依赖关系,当一个对象的状态发生改变时,所有依赖于它的对象都会得到通知并被自动更新。这个模式在很多场景下都有广泛...

    委托和事件-观察者设计模式

    观察者设计模式是一种行为设计模式,它定义了对象之间的一对多依赖关系,当一个对象的状态发生改变时,所有依赖于它的对象都会得到通知并自动更新。 **委托** 委托在.NET中可以视为类型安全的函数指针,它允许我们...

    观察者设计模式 C++实现

    观察者设计模式是一种行为设计模式,它允许一个对象(称为主题)在状态改变时通知其他对象(称为观察者)。这种模式常用于事件处理,当一个对象的状态发生变化时,所有依赖于该状态的对象都会得到通知并自动更新。在...

    观察者设计模式demo

    观察者设计模式是一种行为设计模式,它允许对象在状态改变时通知其他依赖它的对象,而无需显式指定这些依赖关系。在这个"观察者设计模式demo"中,我们可能看到一个具体的实现,特别是在Android开发中的应用,例如在...

    构建者模式、回调和观察者设计模式

    构建者模式、回调和观察者设计模式是软件开发中常用的设计模式,它们分别在不同的场景下发挥着重要的作用。下面将详细介绍这三个设计模式,并通过一个简单的例子来展示它们的应用。 首先,构建者模式(Builder ...

    java中监听机制,以及观察者设计模式

    Java中的监听机制与观察者设计模式是编程中重要的概念,它们在软件开发中扮演着关键的角色,尤其是在构建可扩展和可维护的系统时。这里,我们将深入探讨这两个主题,并通过实例来阐述它们的工作原理和应用。 首先,...

    设计模式之观察者设计模式理解

    轻松理解观察者设计模式

    java观察者设计模式demo源码

    Java观察者设计模式(Observer Design Pattern)是一种行为设计模式,它定义了对象之间的一对多依赖关系,当一个对象的状态发生改变时,所有依赖于它的对象都会得到通知并被自动更新。这种模式常用于事件处理,例如...

    观察者设计模式笔记

    031114_【第11章:Java常用类库】_观察者设计模式笔记

    设计模式--观察者模式java例子

    观察者模式(Observer Pattern)是软件设计模式中的一种行为模式,它定义了对象之间的一对多依赖关系,当一个对象的状态发生改变时,所有依赖于它的对象都会得到通知并自动更新。这种模式常用于事件驱动的系统或者...

    观察者模式java实现观察者设计模式

    观察者模式(Observer Pattern)是软件设计模式中的一种行为模式,它定义了对象之间的一对多依赖关系,当一个对象的状态发生改变时,所有依赖于它的对象都会得到通知并自动更新。在Java中,我们可以利用Java提供的...

    ObserverDemo观察者设计模式Demo

    ObserverDemo观察者设计模式Demo

    跟我一起学 设计模式-观察者模式

    观察者模式,也称为发布-订阅模式或事件驱动模式,是一种行为设计模式,它定义了对象间的一对多依赖关系,当一个对象的状态发生改变时,所有依赖于它的对象都会得到通知并自动更新。 在C#、ASP.NET等.NET框架中,...

    设计模式实现——观察者模式

    观察者模式(Observer Pattern)是软件设计模式中的一种行为模式,它定义了对象之间的一对多依赖关系,当一个对象的状态发生改变时,所有依赖于它的对象都会得到通知并自动更新。这种模式常用于实现发布-订阅...

    观察者模式

    观察者设计模式观察者模式(有时又被称为发布-订阅模式、模型-视图模式、源-收听者模式或从属者模式)是软件设计模式的一种。在此种模式中,一个目标物件管理所有相依于它的观察者物件,并且在它本身的状态改变时...

    设计模式之观察者模式

    观察者模式是软件设计模式中的一种行为模式,它在对象之间定义了一种一对多的依赖关系,当一个对象的状态发生改变时,所有依赖于它的对象都会得到通知并自动更新。这种模式广泛应用于事件驱动编程、发布/订阅系统...

    Java观察者设计模式详解

    Java观察者设计模式是一种行为设计模式,它定义了对象之间的一对多依赖关系,当一个对象的状态发生改变时,所有依赖于它的对象都会得到通知并自动更新。这种模式常用于事件处理系统,将观察者与被观察对象解耦,使得...

Global site tag (gtag.js) - Google Analytics