Spring事件处理

news/2024/11/20 1:39:29/

        在实际业务开发中,有时候复杂性的业务之间需要解耦,常用的方法:同步、异步、MQ。但 MQ 重啊,非必要不提升架构复杂度。

        针对同步和异步使用方式:1.定时器 2.Spring Event.

        Spring Event:

                观察者设计模式,一个 Bean 处理完成任务后希望通知其它 Bean 或者说一个 Bean监听另一个Bean 的行为。

配置异步线程池

package com.lean.event.demo.config;import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.aop.interceptor.AsyncUncaughtExceptionHandler;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.scheduling.annotation.AsyncConfigurer;
import org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor;import java.util.concurrent.Executor;
import java.util.concurrent.ThreadPoolExecutor;@Configuration
public class AsynConfig implements AsyncConfigurer {private static final Logger log= LoggerFactory.getLogger(AsynConfig.class);@Bean("taskAsyncPool")public Executor orderTaskAsyncPool() {ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();executor.setCorePoolSize(10); // 核心线程数executor.setMaxPoolSize(20); // 最大线程数executor.setQueueCapacity(1000); // 队列大小executor.setKeepAliveSeconds(300); // 线程最大空闲时间executor.setThreadNamePrefix("async-Executor-"); // 指定用于新创建的线程名称的前缀。executor.setRejectedExecutionHandler(new ThreadPoolExecutor.CallerRunsPolicy()); // 拒绝策略(一共四种,此处省略)executor.initialize();return executor;}@Overridepublic Executor getAsyncExecutor() {return orderTaskAsyncPool();}// 异常处理器@Overridepublic AsyncUncaughtExceptionHandler getAsyncUncaughtExceptionHandler() {return (ex, method, params) -> log.error(String.format("执行异步任务'%s'", method), ex);}
}

1.同步

import lombok.Getter;
import lombok.Setter;
import lombok.ToString;
import org.springframework.context.ApplicationEvent;
/*** 自定义事件 继承ApplicationEvent*/
@Getter
@Setter
@ToString
public class MessageEvent extends ApplicationEvent {//传输的数据对象private String messageId;public MessageEvent(Object source, String messageId) {super(source);this.messageId = messageId;}
}
package com.lean.event.demo.listener;import com.lean.event.demo.events.MessageEvent;
import lombok.SneakyThrows;
import lombok.extern.slf4j.Slf4j;
import org.springframework.context.ApplicationListener;
import org.springframework.stereotype.Component;
/*** 同步* 监听器:* 实现方式:*      1.实现ApplicationListener*      2.@EventListener*/
@Slf4j
@Component
public class MessageListener implements ApplicationListener<MessageEvent> {@SneakyThrows@Overridepublic void onApplicationEvent(MessageEvent event) {String messageId = event.getMessageId();long start = System.currentTimeMillis();long end = System.currentTimeMillis();log.info("{}:耗时:({})毫秒", messageId, (end - start));}
}

2.异步

| 自定义事件

import lombok.AllArgsConstructor;
import lombok.Data;@Data
@AllArgsConstructor
public class MsgEvent {/*** 该类型事件携带的信息*/public String messageId;
}
import com.lean.event.demo.events.MsgEvent;
import lombok.SneakyThrows;
import lombok.extern.slf4j.Slf4j;
import org.springframework.context.event.EventListener;
import org.springframework.scheduling.annotation.Async;
import org.springframework.stereotype.Component;
@Slf4j
@Component
public class MsgListener {//异步注解 需要开启@EnableAsync异步@Async(value = "taskAsyncPool")@SneakyThrows@EventListener(MsgEvent.class)public void sendMsg(MsgEvent event) {String messageId = event.getMessageId();long start = System.currentTimeMillis();log.info("开发发送短信");log.info("开发发送邮件");long end = System.currentTimeMillis();log.info("{}:发送短信、邮件耗时:({})毫秒", messageId, (end - start));}
}
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.scheduling.annotation.EnableAsync;@EnableAsync
@SpringBootApplication
public class SpringLeanEventApplication {public static void main(String[] args) {SpringApplication.run(SpringLeanEventApplication.class, args);}}

3.测试

package com.lean.event.demo.service;import com.lean.event.demo.events.MsgEvent;
import com.lean.event.demo.events.MessageEvent;
import lombok.Data;
import lombok.extern.slf4j.Slf4j;
import org.springframework.context.ApplicationContext;
import org.springframework.stereotype.Service;
@Slf4j
@Service
@Data
public class MessageService {/** 注入ApplicationContext用来发布事件 */private final ApplicationContext applicationContext;public String testMessage(String messageId) {long start = System.currentTimeMillis();// (同步处理)applicationContext.publishEvent(new MessageEvent(this, messageId));// (异步处理)applicationContext.publishEvent(new MsgEvent(messageId));long end = System.currentTimeMillis();log.info("任务全部完成,总耗时:({})毫秒", end - start);return "购买成功";}
}
package com.lean.event.demo;import com.lean.event.demo.service.MessageService;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;@SpringBootTest
public class MessageServiceTest {@Autowiredprivate MessageService messageService;@Testpublic void buyOrderTest() {messageService.testMessage("123456");}
}

ApplicationEvent: 

类的注释是,被应用事件继承的类,抽象的不能初始化;

 EventObject: 父类

 事件携带一个 Objecgt 对象,可以被发布。

事件监听者,监听到这个事件后,触发自定义逻辑 ( 操作 Object 对象);

注意:

        事件发布后,都会被监听ApplicationEvent的监听对象所监听到。

代码


http://www.ppmy.cn/news/5614.html

相关文章

STM32的ST-link调试下载,各种调试接口硬件介绍

调试原理 STM32F-10X使用M3内核&#xff0c;该内核支持复杂的同i傲视操作&#xff0c;硬件调试模块允许在取指令&#xff08;指令单步运行&#xff09;或访问数据&#xff08;数据断电时&#xff09;使得内核停止。在内核停止时&#xff0c;内核状态都可被查询&#xff0c;完成…

数理统计期末复习笔记(一)

数理统计期末复习笔记 主要内容&#xff1a; 数据压缩&#xff0c;点估计&#xff0c;假设检验&#xff0c;区间检验 Reference: Statistical Inference, Casella&Berger Chapter 6 Data Reduction&#xff08;数据压缩&#xff09; 随机样本&#xff1a; 无限样本&…

记录我の秋招之旅【23届 CV算法岗】

文章目录碎碎念春招实习华为实习魔幻秋招尘埃落定碎碎念 今年(2022年)的秋招不能说"非常困难"吧&#xff0c;只能说是"地狱难度"&#xff0c;相信参与或者从侧面了解过的同学们也能感同身受。从今年的三月份开始着手秋招&#xff0c;期间也一直忙着实验室…

机器学习100天(十六):016 逻辑回归损失函数

机器学习 100 天,今天讲的是:逻辑回归损失函数。 一、如何找到最佳分类直线 讲完了逻辑回归基本原理之后,我们再来思考一个非常关键的问题:就是如何找到最佳的分类直线呢? 如图中所示,如何判断这三条直线哪个更好?线性回归里,我们可以用均方误差作为损失函数,选择均…

MORE CONVNETS IN THE 2020S: SCALING UP KER- NELS BEYOND 51 × 51 USING SPARSITY

论文链接: https://arxiv.org/pdf/2207.03620.pdf code: https://github.com/VITA-Group/SLaKlink MORE CONVNETS IN THE 2020S: SCALING UP KER- NELS BEYOND 51 51 USING SPARSITY一、引言&#xff08;二&#xff09;、大内核注意力&#xff08;二&#xff09;、卷积中的大…

python帮我省下了这笔冤枉钱

背景 今天不得不说一说我这个电脑的事情。我这个电脑是2年前买的&#xff0c;屏幕非常大&#xff0c;是16寸的。 基本上没什么缺点&#xff0c;就是每隔一年&#xff0c;就要处理一下储存问题。 为什么呢&#xff0c;因为我的这台电脑的储存是512G的。所以不是太大。 但是今天…

数据人PK也无人,为什么业务部门的数据需求都是急活?

**导读&#xff1a;**你是不是经常听到数据开发吐槽业务部门&#xff1a;我可以理解业务部门数据需求多&#xff0c;但为什么经常要得这么急呢&#xff1f; 作为一个数据开发者&#xff0c;可以回想一下&#xff0c;当初是怎么进入数据行业的。 是不是也是听一些大V忽悠&…

BEPU物理引擎碰撞系统的架构与设计

前面我们讲解了如何监听物理引擎的碰撞事件, 在物理引擎内核中如何架构与设计碰撞规则,使得物理Entity与周围的物理环境产生碰撞时&#xff0c;如何灵活的控制物理碰撞&#xff0c;本节給大家详细的讲解BEPUphysicsint 物理引擎内部是如何管理与控制碰撞规则的。本文主要讲解3个…