基于【观察者设计模式】设计异步多渠道群发框架

前言:设计模式源于生活

观察者基本概念

观察者模式,又可以称之为发布-订阅模式,观察者,顾名思义,就是一个监听者,类似监听器的存在,一旦被观察/监听的目标发生的情况,就会被监听者发现,这么想来目标发生情况到观察者知道情况,其实是由目标将情况发送到观察者的
白话文:当一个对象发生改变的时候,可以通知其他所有对象

概念很清晰,举个栗子来理解一下观察者模式的含义,我们都在抖音关注了某位大咖的时候,每当这位大咖更新了一条动态时候,关注大咖的粉丝都能收到通知,简单用一张图来表明他们之间的关系

上面这位大咖发布了个动态,然后他的粉丝都收到通知,并知晓了,从这个栗子可以看到,这里包含两类人,一是大咖,二是粉丝,那么翻译到程序中语言就是观察者的主题和观察者

那么大咖就相当于主题,粉丝相当于观察者,随时观察大咖的动态消息,不过大咖也有权力拉黑你或者让你关注,那么从类图的角度了解一下

observer:抽象观察者,是观察者者的抽象类,它定义了一个更新接口,使得在得到主题更改通知时更新自己。这就是我们所有粉丝的抽象
ConcrereObserver:具体观察者,实现抽象观察者定义的更新接口,以便在得到主题更改通知时更新自身的状态。具体每一个粉丝
Subject:抽象主题,他把所有观察者对象保存在一个集合里,可以有任意数量的观察者,抽象主题提供一个接口,可以增加和删除观察者对象。意思就是大咖把所有的粉丝都保存在一个账号里面,粉丝数量不限,可以新增粉丝也可以拉黑粉丝
ConcreteSubject:具体主题,该角色将有关状态存入具体观察者对象,在具体主题的内部状态发生改变时,给所有注册过的观察者发送通知。意思是我们的大咖一有动态,就会把消息给粉丝。

观察者应用场景

1.对一个对象状态的更新,需要其他对象同步更新,而且其他对象的数量动态可变
2.对象仅需要将自己的更新通知给其他对象而不需要知道其他对象的细节
例如:
1.分布式配置中心,当配置发生改变,通过事件监听来刷新配置,常见的有,apollo,nacos,spring config
2.zk的节点,当节点发生变化,会通知所有的客户端
3.多渠道群发,当你关注了某位大咖,那么每次更新动态的时候,所有关注大咖的人,都会收到大咖的动态更新消息提示

我这里通过三种方式,来实现异步多渠道群发框架

三种形式的maven依赖都是一样的

<parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>2.2.1.RELEASE</version>
    </parent>

    <dependencies>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>
        <dependency>
            <groupId>com.google.guava</groupId>
            <artifactId>guava</artifactId>
            <version>29.0-jre</version>
        </dependency>
        <!-- https://mvnrepository.com/artifact/org.projectlombok/lombok -->
        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
            <version>1.18.12</version>
            <!--            <scope>provided</scope>-->
        </dependency>

        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
        </dependency>

    </dependencies>

第一种方式,基于java内存的形式实现

抽象观察者

public interface ObServer {

    /**
     * 发送消息
     *
     * @param msg
     */
    void sendMsg(String msg);
}

短信具体观察者

@Slf4j
public class SmsObServer implements ObServer {

    @Override
    public void sendMsg(String msg) {
        log.info("发送短信消息,内容:{}", msg);
    }
}

邮件具体观察者

@Slf4j
public class EmailObServer implements ObServer {

    @Override
    public void sendMsg(String msg) {
        log.info("发送邮件消息,内容:{}", msg);
    }
}

抽象主题

@Slf4j
public abstract class SunnySubject {

    protected List<ObServer> obServerList = Lists.newArrayList();

    /**
     * 注册观察者
     *
     * @param obServer
     */
    public void addObServer(ObServer obServer) {
        obServerList.add(obServer);
    }

    /**
     * 移除观察者
     *
     * @param obServer
     */
    public void removeObServer(ObServer obServer) {
        boolean contains = obServerList.contains(obServer);
        if (contains) {
            obServerList.remove(obServer);
        }
    }

    /**
     * 通知观察者
     */
    public abstract void notifyObServer(String msg);
}

具体主题

@Slf4j
public class ConcreteSubject extends SunnySubject {

    private ExecutorService executorService;

    public ConcreteSubject() {
        executorService = Executors.newFixedThreadPool(10);
    }

    @Override
    public void notifyObServer(String msg) {
        log.info("目标对象状态已变化......发送通知给观察者中");
        for (ObServer ob : obServerList) {
            executorService.execute(new Runnable() {
                @Override
                public void run() {
                    ob.sendMsg(msg);
                }
            });
        }
    }
}

启动

public class Test {
    public static void main(String[] args) {
        //初始化哦
        SunnySubject sunnySubject = new ConcreteSubject();

        //注册
        sunnySubject.addObServer(new SmsObServer());
        sunnySubject.addObServer(new EmailObServer());

        //通知
        sunnySubject.notifyObServer("你好,观察者");
    }
}

第二种方式,基于SpringIOC容器形式实现

抽象观察者

public interface ObServer {

    /**
     * 发送消息
     *
     * @param msg
     */
    void sendMsg(String msg);
}

短信具体观察者

@Component
@Slf4j
public class SmsObServer implements ObServer {

    @Override
    public void sendMsg(String msg) {
        log.info("发送短信消息,内容:{}", msg);
    }
}

邮件具体观察者

@Component
@Slf4j
public class EmailObServer implements ObServer {

    @Override
    public void sendMsg(String msg) {
        log.info("发送邮件消息,内容:{}", msg);
    }
}

抽象主题

@Slf4j
public abstract class SunnySubject {

    protected List<ObServer> obServerList = Lists.newArrayList();

    /**
     * 注册观察者
     *
     * @param obServer
     */
    public void addObServer(ObServer obServer) {
        obServerList.add(obServer);
    }

    /**
     * 移除观察者
     *
     * @param obServer
     */
    public void removeObServer(ObServer obServer) {
        boolean contains = obServerList.contains(obServer);
        if (contains) {
            obServerList.remove(obServer);
        }
    }

    /**
     * 通知观察者
     */
    public abstract void notifyObServer(String msg);
}

具体主题

@Component
@Slf4j
public class ConcreteSubject extends SunnySubject {

    private ExecutorService executorService;

    public ConcreteSubject() {
        executorService = Executors.newFixedThreadPool(10);
    }

    @Override
    public void notifyObServer(String msg) {
        log.info("目标对象状态已变化......发送通知给观察者中");
        for (ObServer ob : obServerList) {
            executorService.execute(new Runnable() {
                @Override
                public void run() {
                    ob.sendMsg(msg);
                }
            });
        }
    }
}

观察者配置类

@Component
@Slf4j
public class ObServerConfig implements ApplicationRunner {

    @Autowired
    private SmsObServer smsObServer;

    @Autowired
    private EmailObServer emailObServer;

    @Autowired
    private ConcreteSubject concreteSubject;

    @Override
    public void run(ApplicationArguments args) throws Exception {
        concreteSubject.addObServer(smsObServer);
        concreteSubject.addObServer(emailObServer);
    }
}

入口

@RestController
public class ObServerController {

    @Autowired
    private ConcreteSubject concreteSubject;

    @GetMapping("/send")
    public void test(){
        concreteSubject.notifyObServer("你好,观察者");
    }

}

访问:http://localhost:8080/send

当然,如上面我们的代码还可以在优化一下,实现动态注册

@Component
@Slf4j
public class ObServerConfig implements ApplicationRunner, ApplicationContextAware {

    @Autowired
    private SmsObServer smsObServer;

    @Autowired
    private EmailObServer emailObServer;

    @Autowired
    private ConcreteSubject concreteSubject;

    private ApplicationContext applicationContext;

    @Override
    public void run(ApplicationArguments args) throws Exception {
//        concreteSubject.addObServer(smsObServer);
//        concreteSubject.addObServer(emailObServer);

        Map<String, ObServer> map = applicationContext.getBeansOfType(ObServer.class);
        for(String key : map.keySet()){
            ObServer observer = map.get(key);
            concreteSubject.addObServer(observer);
        }
    }

    /**
     * 获取上下文环境对象得到Spring容器中的Bean
     *
     * @param applicationContext
     * @throws BeansException
     */
    @Override
    public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
        this.applicationContext = applicationContext;
    }
}

第三种方式,基于Spring事件形式实现

实体类

public class UserMessageEntity extends ApplicationEvent {

    private String email;

    private Long phone;

    private Long userId;

    public UserMessageEntity(Object source) {
        super(source);
    }

    public UserMessageEntity(Object source, String email, Long phone) {
        super(source);
        this.email = email;
        this.phone = phone;
    }

    @Override
    public String toString() {
        return "UserMessageEntity{" +
                "email='" + email + '\'' +
                ", phone=" + phone +
                ", userId=" + userId +
                '}';
    }
}

邮箱事件回调通知

@Component
@Slf4j
public class EmailListener implements ApplicationListener<UserMessageEntity> {

    @Override
    @Async
    public void onApplicationEvent(UserMessageEntity userMessageEntity) {
        log.info("邮箱通知:{}", userMessageEntity.toString());
    }
}

短信事件回调通知

@Component
@Slf4j
public class SmsListener implements ApplicationListener<UserMessageEntity> {

    @Override
    public void onApplicationEvent(UserMessageEntity userMessageEntity) {
        log.info("短信通知:{}", userMessageEntity.toString());
    }
}

入口

@RestController
public class ObServerController {

    @Autowired
    private ApplicationEventPublisher applicationEventPublisher;

    @GetMapping("/send")
    public void sendTwo() {
        UserMessageEntity userMessageEntity = new UserMessageEntity(this, "123456@163.com", 15096111111L);
        applicationEventPublisher.publishEvent(userMessageEntity);
    }
}

访问:http://localhost:8080/send

总结

观察者模式的主要优点在于可以实现表示层和数据逻辑层的分离,并在观察目标和观察者之间建立一个抽象的耦合,支持广播通信;其主要缺点在于如果一个观察目标对象有很多直接和间接的观察者的话,将所有的观察者都通知到会花费很多时间,而且如果在观察者和观察目标之间有循环依赖的话,观察目标会触发它们之间进行循环调用,可能导致系统崩溃。
其实还有一点需要我们去了解,在上面的例子当中我们的会发现,其实粉丝的消息是大咖推过来的,还有一种观察者模式,也就是我们的粉丝主动去获取消息。
(1)推模型: 主题对象向观察者推送主题的详细信息,不管是否需要。
(2)拉模型:主题对象在通知观察者的时候,只传递少量信息。如果观察者需要更具体的信息,由观察者主动到主题对象中获取

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