springboot中应用观察者模式

近几天在学习springboot源码时发现,在springboot项目启动过程中设置了很多监听器,发现监听器的原理和设计模式中的 观察者模式 很相似。故灵感来袭总结一下 springboot中应用观察者模式

观察者模式解决的问题是:当一个东西(后文称之为 前者)的状态发生改变时,一些与其关联的东西(后文称之为 后者)也会随之做出各自改变。传统的做法是前者在变化后需要直接遍历一个个的通知(调用)到后者,当以后有新的后者需要扩展时,前者的代码也需要改动,不利于扩展。观察者模式解决的问题是,利用java的多态特性,抽象前者与后者直接耦合,在代码层次,前者在改变时只需要发出通知即可,而不需要关心具体通知到谁,后者只需要在前者中注册即可。这样做方便了扩展,当以后有新的后者需要扩展时,直接新增就行了,不需要改动前者的代码。

具体实现demo:

       利用springboot和java的util包中定义的观察者与被观察者工具类实现

1,先定义被观察者

package com.kun.watcher.observed;

import com.kun.watcher.NotifyType;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;

import javax.annotation.PostConstruct;
import java.util.List;
import java.util.Observable;
import java.util.Observer;
import java.util.logging.Logger;

/**
 * 接口被观察者
 *
 * @author kun
 * @date 2020/08/04 15:27
 */
@Component
public class WebObserved extends Observable {

    private static final Logger LOG = Logger.getGlobal();

    @Autowired
    private List<Observer> observerList;

    @PostConstruct
    public void observerRegister() {
        LOG.info("注册观察者-->");
        observerList.forEach(this::addObserver);
    }

    public void useWeb() {
        try {
            LOG.info("开始调用web了");
            this.setChanged();
            this.notifyObservers();

            Thread.sleep(2000);
            LOG.info("web被调用了2秒了");
            this.setChanged();
            this.notifyObservers(NotifyType.KUN);

            Thread.sleep(2000);
            LOG.info("web被调用了4秒了");
            this.setChanged();
            this.notifyObservers(NotifyType.ZHANGSAN);

            Thread.sleep(2000);
            LOG.info("web被调用了6秒了");
            this.setChanged();
            this.notifyObservers(NotifyType.LISI);

            Thread.sleep(2000);
            LOG.info("web被调用结束");
            this.setChanged();
            this.notifyObservers();

        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }

}

继承Observable被观察者类,并使用@Component注解将我们自定义的被观察者类交给spring管理。注入所有观察者对象,

利用@PostConstruct注解,在此被观察者的bean初始化完成时调用方法,将所有观察者对象注册进Observable的列表

在自定义的方法useWeb()中在合适的时机发出通知(在调用notifyObservers()方法之前一定要先调用setChanged()方法,将状态置为改变,然后通知才会有效)

2,定义观察者

package com.kun.watcher.observer;

import com.kun.watcher.NotifyType;
import com.kun.watcher.observed.WebObserved;
import org.springframework.stereotype.Component;

import java.util.Observable;
import java.util.Observer;
import java.util.logging.Logger;

/**
 * 观察者昆
 *
 * @author kun
 * @date 2020/08/04 16:13
 */
@Component
public class KunObserver implements Observer {

    private static final Logger LOG = Logger.getGlobal();

    @Override
    public void update(Observable o, Object arg) {
        if (arg == null || (o instanceof WebObserved && NotifyType.KUN.toString().equals(((NotifyType) arg).name()))) {
            LOG.info("昆收到通知");
        }
    }
}

package com.kun.watcher.observer;

import com.kun.watcher.NotifyType;
import com.kun.watcher.observed.WebObserved;
import org.springframework.stereotype.Component;

import java.util.Observable;
import java.util.Observer;
import java.util.logging.Logger;

/**
 * 观察者张三
 *
 * @author kun
 * @date 2020/08/04 16:13
 */
@Component
public class ZhangsanObserver implements Observer {

    private static final Logger LOG = Logger.getGlobal();

    @Override
    public void update(Observable o, Object arg) {
        if (arg == null || (o instanceof WebObserved && NotifyType.ZHANGSAN.toString().equals(((NotifyType) arg).name()))) {
            LOG.info("张三收到通知");
        }
    }
}

package com.kun.watcher.observer;

import com.kun.watcher.NotifyType;
import com.kun.watcher.observed.WebObserved;
import org.springframework.stereotype.Component;

import java.util.Observable;
import java.util.Observer;
import java.util.logging.Logger;

/**
 * 观察者李四
 *
 * @author kun
 * @date 2020/08/04 16:13
 */
@Component
public class LisiObserver implements Observer {

    private static final Logger LOG = Logger.getGlobal();

    @Override
    public void update(Observable o, Object arg) {
        if (arg == null || (o instanceof WebObserved && NotifyType.LISI.toString().equals(((NotifyType) arg).name()))) {
            LOG.info("李四收到通知");
        }
    }
}

package com.kun.watcher.observer;

import com.kun.watcher.NotifyType;
import com.kun.watcher.observed.WebObserved;
import org.springframework.stereotype.Component;

import java.util.Observable;
import java.util.Observer;
import java.util.logging.Logger;

/**
 * 观察者王二麻子
 *
 * @author kun
 * @date 2020/08/04 16:13
 */
@Component
public class WangermaziObserver implements Observer {

    private static final Logger LOG = Logger.getGlobal();

    @Override
    public void update(Observable o, Object arg) {
        if (arg == null || (o instanceof WebObserved && NotifyType.WANGERMAZI.toString().equals(((NotifyType) arg).name()))) {
            LOG.info("王二麻子收到通知");
        }
    }
}

定义了几个观察者,实现Observer观察者接口,并交给spring容器,这样就可以实现自动注册。

3,定义一个controller,前端发送http请求,改变了被观察者的状态,通知所有观察者。

package com.kun.watcher.controller;

import com.kun.watcher.observed.WebObserved;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.ResponseBody;

/**
 * 接口
 *
 * @author kun
 * @date 2020/08/04 15:24
 */
@Controller
public class Web {

    @Autowired
    private WebObserved webObserved;

    @GetMapping("/b")
    @ResponseBody
    public String getBegin(){
        webObserved.useWeb();
        return "success";
    }
}

启动,发送请求,结果如下:

2020-08-05 11:12:28.908  INFO 2116 --- [nio-8001-exec-1] global  : 开始调用web了
2020-08-05 11:12:28.908  INFO 2116 --- [nio-8001-exec-1] global  : 张三收到通知
2020-08-05 11:12:28.908  INFO 2116 --- [nio-8001-exec-1] global  : 王二麻子收到通知
2020-08-05 11:12:28.908  INFO 2116 --- [nio-8001-exec-1] global  : 李四收到通知
2020-08-05 11:12:28.908  INFO 2116 --- [nio-8001-exec-1] global  : 昆收到通知
2020-08-05 11:12:30.910  INFO 2116 --- [nio-8001-exec-1] global  : web被调用了2秒了
2020-08-05 11:12:30.910  INFO 2116 --- [nio-8001-exec-1] global  : 昆收到通知
2020-08-05 11:12:32.911  INFO 2116 --- [nio-8001-exec-1] global  : web被调用了4秒了
2020-08-05 11:12:32.911  INFO 2116 --- [nio-8001-exec-1] global  : 张三收到通知
2020-08-05 11:12:34.911  INFO 2116 --- [nio-8001-exec-1] global  : web被调用了6秒了
2020-08-05 11:12:34.911  INFO 2116 --- [nio-8001-exec-1] global  : 李四收到通知
2020-08-05 11:12:36.912  INFO 2116 --- [nio-8001-exec-1] global  : web被调用结束
2020-08-05 11:12:36.912  INFO 2116 --- [nio-8001-exec-1] global  : 张三收到通知
2020-08-05 11:12:36.912  INFO 2116 --- [nio-8001-exec-1] global  : 王二麻子收到通知
2020-08-05 11:12:36.912  INFO 2116 --- [nio-8001-exec-1] global  : 李四收到通知
2020-08-05 11:12:36.913  INFO 2116 --- [nio-8001-exec-1] global  : 昆收到通知

若要扩展观察者,仅需要实现Observer接口,并加上@Component注解即可

  • 1
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 7
    评论
在Spring Boot,可以使用Java内置的观察者模式来实现事件驱动的编程。具体步骤如下: 1. 定义观察者接口:通常情况下,观察者接口包含一个update()方法,用于接收事件通知。 ```java public interface Observer { void update(); } ``` 2. 定义事件类:通常情况下,事件类包含事件类型和事件数据。 ```java public class Event { private EventType eventType; private Object eventData; // getters and setters } ``` 3. 定义事件类型枚举类:用于区分不同的事件类型。 ```java public enum EventType { EVENT_TYPE_1, EVENT_TYPE_2 // ... } ``` 4. 定义观察者实现类:实现Observer接口,并在update()方法处理事件。 ```java public class MyObserver implements Observer { @Override public void update() { // 处理事件 } } ``` 5. 定义事件发布类:用于发布事件。 ```java @Component public class EventPublisher { @Autowired private List<Observer> observers; public void publish(Event event) { for (Observer observer : observers) { observer.update(event); } } } ``` 6. 在需要监听事件的地方注册观察者:使用@Autowired注入EventPublisher,并调用其publish()方法发布事件。 ```java @Component public class MyService implements Observer { @Autowired private EventPublisher eventPublisher; @PostConstruct public void init() { eventPublisher.register(this); } @Override public void update(Event event) { // 处理事件 } } ``` 以上就是使用Spring Boot提供的观察者模式的基本步骤。需要注意的是,观察者模式适用于一对多的场景,即一个事件可以被多个观察者接收和处理。在实际应用,也可以根据具体的业务需求进行扩展和改进。
评论 7
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值