技术交流QQ群【JAVA,C++,Python,.NET,BigData,AI】:170933152

这篇博客写的不错啊,在springcloud 中使用事件驱动,有时候是很有必要的

文章目录

1.spring下使用event模型
1.1 定义event
1.2 event的监听处理类。监听类实现ApplicationListener 里onApplicationEvent方法即可
1.3 发布事件
2.evnet模型的注意点
3. 一种更优雅的方式——@EventListener
3.1 发布事件
3.2 定义事件源
3.3 监听事件@EventListener
3.4 监听事件时的事务隔离

我们知道观察者模式可以实现代码的解耦,而spring的event模型就是这种设计模式的极佳体现。一个事件包含:事件发布、监听、和事件源。在spring中我们可以通过ApplicationContext的publishEvent方法去发布事件;通过实现ApplicationListener接口来自定义自己的监听器;继承ApplicationEvent类来实现事件源。下面以一个实例来说明:
1.spring下使用event模型

1.1 定义event

/**
 * event的基类
 *
 * @author 94977
 * @create 2018/7/22
 */
public abstract class BaseEvent extends ApplicationEvent {
    public BaseEvent(Object source) {
        super(source);
    }
}

public class FaceEvent extends BaseEvent {

/**
    * @author 94977
    * @time 2018/7/22 15:50
    * @param * @param null
    * @return
    * @description  必须要实现的构造方法
    */
    public FaceEvent(User user) {
        super(user);
    }
}

1.2 event的监听处理类。监听类实现ApplicationListener 里onApplicationEvent方法即可

@Component
public class FaceEventListener implements ApplicationListener {

@Override
    public void onApplicationEvent(ApplicationEvent event) {
        if(event instanceof FaceEvent){
            User user = (User) event.getSource();
            LOGGER.info("===> 收到人脸事件:  {}",user);
            // .....
            System.out.println("人脸事件处理结束。。。");
        }
    }
}

当然通过event instanceof FaceEvent判断事件源来处理的方式不是很优雅。有更好的方式,接口ApplicationListener支持泛型,可以通过泛型来判断处理的事件源。如下只处理FaceEvent源。

@Component
public class FaceEventListener extends BaseEventListener implements ApplicationListener<FaceEvent> {

@Override
    public void onApplicationEvent(FaceEvent event) {
        User user = (User) event.getSource();
        LOGGER.info("===> 收到人脸事件:  {}",user);
        // .....
        System.out.println("人脸事件处理结束。。。");

}
}

如果要实现有序的监听,实现SmartApplicationListener 接口即可

1.3 发布事件

@Service
public class FaceHandler {

@Autowired
    private ApplicationContext applicationContext;

public void handle(){
        User user = new User();
        user.setAge(34);
        user.setUsername("人脸事件");
        user.setHobby("抓拍");
        //发布事件
        applicationContext.publishEvent(new FaceEvent(user));
        //进行其他业务处理
    }

以上即可。

2.evnet模型的注意点

事件没要处理的监听器,就会被抛弃。
一个事件可以同时被多个监听处理类监听处理。
以上处理事件都是同步的,如果发布事件处的业务存在事务,监听器处理也会在相同的事务中。这个一定要注意!如果对于事件的处理不想受到影响,可以onApplicationEvent方法上加@Aync支持异步(参考taskExecutor的使用)。
原理部分可以参考 博客 事件体系
3. 一种更优雅的方式——@EventListener

3.1 发布事件

我们可以通过工具类发布来避免在代码耦合注入ApplicationContext,工具类实现ApplicationEventPublisherAware 接口,具体可参考spring的aware学习。
       这里有一个小细节,如果通过注入ApplicationContext的方式来发布事件,idea在代码左边会有一个类似耳机的小图标,点击可以跳到监听此发布事件的监听者位置,用工具类发布事件就没有此提示了。

3.2 定义事件源

public abstract class BaseEvent<T> extends ApplicationEvent {

private static final long serialVersionUID = 895628808370649881L;

protected T eventData;

public BaseEvent(Object source, T eventData){
        super(source);
        this.eventData = eventData;
    }

public BaseEvent(T eventData){
        super(eventData);
    }
    
    public T getEventData() {
        return eventData;
    }
    public void setEventData(T eventData) {
        this.eventData = eventData;
    }
}

需要发布的事件继承此BaseEvent

public class FaceEvent extends BaseEvent<User> {
    
    public FaceEvent(User user) {
        super(user);
    }

public FaceEvent(Object source, User user){
        super(source,user);
    }

}

如果代码结构较复杂,多处发布相同的事件,建议发布事件时将this作为source传递,便于通过分析日志确定发布源。

3.3 监听事件@EventListener

在spring4.2中我们可以以更加简洁的方式来监听event的发布,监听事件我们不必再实现ApplicationListener接口了,只要在方法上添加注解@EventListener即可:

@EventListener
    public void onApplicationEvent(FaceEvent event) {
        User user = (User) event.getSource();
        String name = Thread.currentThread().getName();
        LOGGER.info("===> 收到人脸事件:  {},线程名为: {}",user,name);
    }

会根据方法参数类型来自动监听相应事件的发布。
       如果要监听多个事件类型的发布,可以在@EventListener(classes = {FaceEvent.class,ArmEvent.class})指定,spring会多次调用此方法来处理多个事件。但是注意此时,方法参数不能有多个,否则会发生转换异常,可以将使用多个事件的父类作为唯一的方法参数来接收处理事件,但除非必要否则并不推荐监听多个事件的发布。

如果有多个监听器监听同一事件,我们可以在方法上使用spring的@order注解来定义多个监听器的顺序,如:

@EventListener
    @Order(4)
    public void onApplicationEvent(FaceEvent event) {
        User user = (User) event.getSource();
        LOGGER.info("===> A 收到人脸事件:  {}",user);
    }

@EventListener({FaceEvent.class,ArmEvent.class})
    @Order(3)
    public void onApplicationEvent3(Object event) {

if(event instanceof FaceEvent){
            LOGGER.info("===> B 收到人脸事件:  {}",((FaceEvent) event).getEventData());
        }else if(event instanceof ArmEvent){
            ArmEvent armEvent = (ArmEvent) event;
            LOGGER.info("===> B 收到臂膀事件:  {}",armEvent.getEventData());
        }
    }

这真的是很方便。

@EventListener还有一个属性,condition()里可以使用SPEL表达式来过滤监听到事件,即只有符合某种条件的才进行接收处理。暂时还用不到。
3.4 监听事件时的事务隔离

@TransactionalEventListener和@EventListener都可以监听事件,但前者可以对发布事件和监听事件进行一些事务上的隔离。@TransactionalEventListenerr指不和发布事件的方法在同一个事务内,发布事件的方法事务结束后才会执行本监听方法,监听逻辑内发生异常不会回滚发布事件方法的事务。

@Transactional(rollbackFor = Exception.class)
    public void handle(){
        User user = new User();
        user.setAge(34);
        user.setUsername("人脸事件");
        user.setHobby("抓拍");

//处理完上面的逻辑后,发布事件
        EventPublisherUtil.publishEvent(new FaceEvent(user));
 
        //数据库添加操作
        Integer integer = deviceAlarmService.addDevice();
    }

可以看到发布事件的方法处在事务控制中,我们使用@TransactionalEventListener来监听事件:

@TransactionalEventListener(fallbackExecution = true)
    public void onApplicationEvent(FaceEvent event) {
        User user = event.getEventData();
        LOGGER.info("===> A 收到人脸事件:  {}}",user);

//@TransactionalEventListener指不和发布事件的在同一个事务内,发布事件的方法事务结束后才会执行本方法,本方法发生异常不会回滚发布事件的事务,
        throw new  RuntimeException("监听事件抛出异常");
    }

运行结果,addDevice正常在数据库插入数据,但是修改为@EventListener监听则插入数据失败。

@TransactionalEventListener有一个属性为fallbackExecution,默认为false,指发布事件的方法没有事务控制时,监听器不进行监听事件,此为默认情况! fallbackExecution=true,则指发布事件的方法没有事务控制时,监听方法仍可以监听事件进行处理。
    /**
     * Whether the event should be processed if no transaction is running.
     */
    boolean fallbackExecution() default false;

刚才我们说到使用@TransactionalEventListener会在发布事件的方法事务结束后执行监听方法,但其实我们还可以进行细化的控制。它有一个属性为TransactionPhase,默认为TransactionPhase.AFTER_COMMIT,即事务提交后。还可以根据需要选择AFTER_COMPLETION、BEFORE_COMMIT、AFTER_ROLLBACK。
       但仍需注意,如果fallbackExecution=false,且发布事件的方法没有事务控制时,监听器根本不会监听到事件,此处的TransactionPhase也就没有意义了。
---------------------

SpringCloud工作笔记078---SpringBoot中使用sping事件驱动模型相关推荐

  1. 【学习笔记】springBoot中获取sping管理的bean

    文章目录 一.使用场景 二.springBoot中获取sping管理的bean 2.1 生成工具类SpringContextUtil 2.2 使用工具类SpringContextUtil 2.3 注意 ...

  2. SpringCloud工作笔记085---SpringBoot项目中防止跨站脚本攻击功能添加

    技术交流QQ群[JAVA,C++,Python,.NET,BigData,AI]:170933152 一:什么是XSS XSS攻击全称跨站脚本攻击,是一种在web应用中的计算机安全漏洞,它允许恶意we ...

  3. SpringCloud工作笔记048---RESTful API 中 HTTP 状态码的定义_以及把RESTFul版本号_放到http协议header中_以及RestFul设计时的两个误区

    JAVA技术交流QQ群:170933152 ------------------------- RESTful架构有一些典型的设计误区. 最常见的一种设计错误,就是URI包含动词.因为"资源 ...

  4. SpringCloud工作笔记034---Zuul配置项中sensitiveHeaders和ignoredHeaders的不同

    记录一下: JAVA技术交流QQ群:170933152 sensitiveHeaders会过滤客户端附带的headers 例如: sensitiveHeaders: X-ABC 如果客户端在发请求是带 ...

  5. SpringCloud工作笔记084---SpringCloud项目中,关于防止表单提交_使用redis+Aspect面向切面实现

    技术交流QQ群[JAVA,C++,Python,.NET,BigData,AI]:170933152 这里用注解+redis的方式来防止表单提交 先声明注解: package cn.gov.credr ...

  6. SpringBoot学习笔记(4)----SpringBoot中freemarker、thymeleaf的使用

    1. freemarker引擎的使用 如果你使用的是idea或者eclipse中安装了sts插件,那么在新建项目时就可以直接指定试图模板 如图: 勾选freeMarker,此时springboot项目 ...

  7. SpringBoot学习笔记(9)----SpringBoot中使用关系型数据库以及事务处理

    在实际的运用开发中,跟数据库之间的交互是必不可少的,SpringBoot也提供了两种跟数据库交互的方式. 1. 使用JdbcTemplate 在SpringBoot中提供了JdbcTemplate模板 ...

  8. 利用Quartz设计采集系统并实现系统双活机制_在SpringCloud中自己设计系统双活---SpringCloud工作笔记178

    因为项目仅仅是不停的抓数据,没有弄成分布式的,但依然需要系统双活来保证系统稳定. 这个时候,我的思路是:这个在另一篇博文中有些的详细思路(在Timer定时任务中_基于Redis自己实现一套双机互备_双 ...

  9. springcloud工作笔记107---Springboot中MyBatis 自动转换 map-underscore-to-camel-case=true 开启驼峰命名映射

    技术交流QQ群[JAVA,C++,Python,.NET,BigData,AI]:170933152 注意只有开启了这个配置,mybatis才会自动映射,但是即使开启了,也需要注意, 该配置的也要配置 ...

最新文章

  1. O太多,具体都代表什么呢?
  2. UMDF驱动开发入门
  3. Hadoop Pig学习笔记 各种SQL在PIG中实现
  4. html图像特征提取,图像识别之图像特征提取
  5. [PA 2014]Kuglarz
  6. R语言在C#使用DCom中遇到的若干问题
  7. ln命令:软链接建立与删除
  8. 论文阅读笔记(十)——Acoustic Scene Classification Using Reduced MobileNet Architecture
  9. Python 内置函数详解
  10. SAP ABAP BAPI_MATERIAL_AVAILABILITY 查询可用库存
  11. 通信管线及宽带接入工程建设中主要涉及的 设计、施工及验收规范
  12. 三菱PLC控制东芝4轴机器人程序,有完整的PLC程序带注释, 结构清楚,信捷触摸屏程序,电气图纸,东芝机械手程序,适合你学习应用,附赠东芝机器人编程软件,可以模拟运行。
  13. 另类QQ客户端 直接用QQ号登陆 (借花献佛)
  14. python建站与java建站有何不同_Python与JAVA有何区别?
  15. Java学习笔记Day06 工具类及常用算法
  16. imx6ul linux读取DS18B20温度
  17. 公开密钥基础设施PKI
  18. oracle查询营业执照不对的sql
  19. 超好用的网页版组态编辑器BY组态
  20. 微信公众平台邮箱修改方法

热门文章

  1. Nginx——反向代理多个服务器
  2. HDU-2553N皇后问题(dfs)
  3. 和最大子序列(dp)
  4. 我心目中的Asp.net核心对象
  5. 远程LInux和秘钥认证
  6. 关于ecplise中一些很实用的技巧
  7. TextView跑步灯效果及在特殊情况下无效的解决方式
  8. nginx从0到1之参数配置
  9. Odoo链接magento纪实
  10. BI的需求调研的方法分类