TransactionalEventListener,事务事件监听,可监听事务提交前、提交后、事务回滚、事务完成(成功或失败)。
Spring内置了简便的事件机制,可以非常方便的实现事件驱动,核心类包括
ApplicationEvent,具体事件内容,事件抽象基类,可继承该类自定义具体事件ApplicationEventPublisher,事件发布器,可以发布ApplicationEvent,也可以发布普通的Object对象ApplicationListener,事件监听器,可以使用注解@EventListenerTransactionalEventListener,事务事件监听,可监听事务提交前、提交后、事务回滚、事务完成(成功或失败)public class UserEntity { private long id; private String name; private String msg;}
可以添加条件condition,限制监听具体的事件
(资料图片)
@Slf4j@Componentpublic class RegisterListener { @EventListener(condition = "#entity.id != null and #entity.async==false ") public void handlerEvent(UserEntity entity) { try { // 休眠5秒 TimeUnit.SECONDS.sleep(5); } catch (InterruptedException e) { e.printStackTrace(); } log.info("handlerEvent: {}", entity); }}
public interface IRegisterService { public void register(String name);}
@Servicepublic class RegisterServiceImpl implements IRegisterService { @Resource private ApplicationEventPublisher applicationEventPublisher; @Override public void register(String name) { UserEntity entity = new UserEntity(); entity.setName(name); entity.setId(1L); entity.setMsg("新用户注册同步调用"); applicationEventPublisher.publishEvent(entity); }}
@Slf4j@Controllerpublic class TestController { @Resource private IRegisterService registerService; @RequestMapping("test") @ResponseBody public void test1(String name) { registerService.register(name); log.info("执行同步调用结束"); }}
在浏览器中输入地址:http://localhost/test?name=nik
控制台输出:
handlerEvent: UserEntity(id=1, name=nik, msg=新用户注册同步调用)执行同步调用结束
@Async@EventListener(condition = "#entity.name != null and #entity.async ")public void handlerEventAsync(UserEntity entity) { try { TimeUnit.SECONDS.sleep(5); } catch (InterruptedException e) { e.printStackTrace(); } log.info("handlerEventAsync: {}", entity);}
@Overridepublic void registerAsyn(String name) { UserEntity entity = new UserEntity(); entity.setName(name); entity.setId(1L); entity.setMsg("新用户注册异步调用"); entity.setAsync(true); applicationEventPublisher.publishEvent(entity);}
@RequestMapping("test")@ResponseBodypublic void test(String name) { registerService.registerAsyn(name); log.info("执行异步调用结束");}
控制台输出:
执行异步调用结束handlerEventAsync: UserEntity(id=1, name=nik, msg=新用户注册异步调用)
比如,用户注册成功后给用户发送成功短信,那么注册成功必然是注册方法事务提交成功后才代表成功。
Spring提供了注解@TransactionalEventListener监听事务事件,在@EventListener基础上增加了属性phase,包含以下四个值:
AFTER_COMMIT,事务提交成功后,默认BEFORE_COMMIT,事务提交前AFTER_ROLLBACK,事务回滚后AFTER_COMPLETION,事务完成,AFTER_COMMIT或AFTER_ROLLBACKpublic class RegisterCommitEvent extends ApplicationEvent { @Getter @Setter private String msg; @Getter @Setter private String name; public RegisterCommitEvent(UserEntity source) { super(source); this.msg = source.getMsg(); this.name = source.getName(); }}
@Override@Transactionalpublic void registerCommit(String name) { UserEntity entity = new UserEntity(); entity.setName(name); entity.setMsg("新用户注册事务提交事件"); RegisterCommitEvent registerEvent = new RegisterCommitEvent(entity); userDao.save(entity); // 发送事件 applicationEventPublisher.publishEvent(registerEvent);}
@Async@TransactionalEventListener(phase = TransactionPhase.AFTER_COMMIT)public void handlerEventCmmit(RegisterCommitEvent event) { try { TimeUnit.SECONDS.sleep(5); } catch (InterruptedException e) { e.printStackTrace(); } log.info("handlerEventCmmit: {}", event);}
@RequestMapping("test")@ResponseBodypublic void test(String name) { registerService.registerCommit(name); log.info("执行事务调用结束");}
控制台输出:
执行事务调用结束handlerEventCmmit: RegisterCommitEvent[source=UserEntity(id=0, name=nik, msg=新用户注册事务提交事件)]
Spring ApplicationEvent事件处理机制使用起来简单方便,可以对程序进行有效解耦。
虽然可以发送任意类型的对象,但是在实际业务中容易产生混乱,建议根据实际业务,定义好各类事件,并在监听方法中实现异步处理。