项目实战—观察者模式(自定义高性能的订阅-发布模型)

观察者模式是对象的行为模式,又叫发布-订阅(Publish/Subscribe)模式、模型-视图(Model/View)模式、源-监听器(Source/Listener)模式或从属者(Dependents)模式。

观察者模式定义了一种一对多的依赖关系,让多个观察者对象同时监听某一个主题对象。这个主题对象在状态上发生变化时,会通知所有观察者对象,使它们能够自动更新自己。

1. JDK提供的观察者模式

在JAVA语言的java.util库里面,提供了一个Observable类以及一个Observer接口,构成JAVA语言对观察者模式的支持。

事件对象:

@Data
public class NameEvent {

    private String name;

    //信息
    private String message;
}

发布者(被观察者):

import java.util.Observable;
import java.util.Observer;

public class BusService {

    //发布者对象
    private NameWatched observable = new NameWatched();
    
    public void addObservers(Observer... observers) {
        //支持动态新增观察者
        for (Observer observer : observers) {
            observable.addObserver(observer);
        }
    }

    /**
     * 业务方法
     */
    public void bus(String name) {
        //当名字为tom时,触发事件
        if ("tom".equals(name)) {
            NameEvent event = new NameEvent();
            event.setName("tom");
            event.setMessage("触发事件啦!");
            observable.sendNameEvent(event);
        }

    }

    /**
     * 发布者对象(观察者)
     */
    public static class NameWatched extends Observable{

        public void sendNameEvent(NameEvent nameEvent){
            //发送变化了
            setChanged();
            //推送消息
            notifyObservers(nameEvent);
        }

    }

}

订阅者(观察者):

@Slf4j
public class NameWatcher implements Observer {
    @Override
    public void update(Observable o, Object arg) {
        //获取到业务参数
      log.info("监听者-打印数据{}", JSON.toJSONString(arg));
    }
}

测试代码:

public class Testlucene {
    public static void main(String[] args) {

        //线程1,执行业务逻辑
        new Thread(()->{
            BusService busService=new BusService();
            busService.addObservers(new NameWatcher());
            busService.bus("tom");
        }).start();


        try {
            Thread.sleep(10000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}

结果:

21:15:29.258 [Thread-0] INFO com.tellme.test.NameWatcher - 监听者-打印数据{"message":"触发事件啦!","name":"tom"}

缺点:

  1. 传递参数的方式在notifyObservers加了sync(this)关键字,会严重影响性能,但作用仅仅就是同步获取增删的订阅者。
  2. 传递this对象,不能声明全局的Observable对象。
  3. 多个订阅者和发布者时同一个线程。会影响发布者性能!

2. Google Guava的EventBus提供的观察者模式

使用方式:
JAVA进阶篇(10)—Guava实现的EventBus(观察者模式)

缺点:

  1. EventBus需要注意:发布者和订阅者使用同一个线程,可能会影响发布者的性能。但可以保证单线程中事件的发布顺序和调度顺序保持一致。
  2. AsyncEventBus需要注意的是:发布者和订阅者可以使用不同的线程处理;发布事件时维护了一个LinkedQueue,若订阅者消费速度慢,可能会造成内存溢出;采用全局队列维护事件顺序性,但不能完全保证调度和发布的顺序;性能不如直接分发好;
  3. guava的EventBus虽然通过注解的方式更加灵活,但是没有接口的语法层面的依赖关系,代码维护性、可读性不是特别好。

3. 自定义实现

public class SealObservable {

    /**
     * 线程安全,且适合读多写少的场景
     */
    private List<SealObserver> sealObservers = new CopyOnWriteArrayList<>();

    private Executor executor;

    public SealObservable() {
        executor = DirectExecutor.INSTANCE;
    }

    /**
     * @param sync true:使用同一个线程处理消息。false:开启异步线程处理消息
     */
    public SealObservable(boolean sync) {
        if (sync) {
            executor = DirectExecutor.INSTANCE;
        } else {
            executor = new ThreadPoolExecutor(4,
                    8, 60, TimeUnit.SECONDS,
                    new ArrayBlockingQueue<>(10), new ThreadPoolExecutor.DiscardPolicy());
        }
    }

    public SealObservable(Executor executor) {
        this.executor = executor;
    }

    public void addObserver(SealObserver o) {
        if (o == null)
            throw new NullPointerException();
        if (!sealObservers.contains(o)) {
            sealObservers.add(o);
        }
    }

    public void deleteObserver(SealObserver o) {
        sealObservers.remove(o);
    }

    /**
     * 通知消息
     */
    public void notifyObservers(Object arg) {
        //没有监听者,快速失败
        if (sealObservers.size() == 0) {
            return;
        }
        //开启线程配置
        for (SealObserver sealObserver : sealObservers) {
            executor.execute(() -> {
                sealObserver.notice(arg);
            });
        }
    }

}

直接线程池:

/**
 * 直接线程池。
 */
public enum DirectExecutor implements Executor {
    INSTANCE;

    @Override
    public void execute(Runnable command) {
        command.run();
    }

    @Override
    public String toString() {
        return "SealExecutors.directExecutor()";
    }
}

观察者对象:

public interface SealObserver {

    /**
     * 触发事件调用的方法
     *
     * @param arg 事件对象
     */
    void notice(Object arg);
}
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 205,236评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,867评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,715评论 0 340
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,899评论 1 278
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,895评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,733评论 1 283
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,085评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,722评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,025评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,696评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,816评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,447评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,057评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,009评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,254评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,204评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,561评论 2 343

推荐阅读更多精彩内容