在實際項目中,我們往往需要自定義一些事件和監(jiān)聽器來滿足業(yè)務(wù)場景,比如在微服務(wù)中會有這樣的場景:
服務(wù) A 在處理完某個邏輯之后,需要通知微服務(wù) B 去處理另一個邏輯,或者微服務(wù) A 處理完某個邏輯之后,需要將數(shù)據(jù)同步到微服務(wù) B,這種場景非常普遍,這個時候,我們可以自定義事件以及監(jiān)聽器來監(jiān)聽,一旦監(jiān)聽到微服務(wù) A 中的某事件發(fā)生,就去通知微服務(wù) B 處理對應(yīng)的邏輯。
步驟1, 自定義事件A
public class MyEvent extends ApplicationEvent {
private User user;
public MyEvent(Object source, User user) {
super(source);
//這里寫服務(wù)A的內(nèi)容,如下
this.user = user;
ystem.out.println("A服務(wù)完成");
}
public User getUser() {
return user;
}
}步驟2, 監(jiān)聽事件A,在監(jiān)聽事件中 實現(xiàn)B或者B
@Component
public class MyEventListener implements ApplicationListener<MyEvent> {
public void onApplicationEvent(MyEvent myEvent) {
User user = myEvent.getUser();
// 一旦MyEvent事件發(fā)生,這里就會執(zhí)行
// 處理事件,實際項目中可以通知別的微服務(wù)或者處理其他邏輯等等
System.out.println("用戶名:" + user.getUsername());
System.out.println("密碼:" + user.getUserpwd());
System.out.println("監(jiān)聽事件發(fā)生后, 通知的服務(wù)已經(jīng)完成");
}
}步驟3,發(fā)布事件
@Service
public class UserService {
@Resource
private ApplicationContext applicationContext;
/**
* 發(fā)布事件
* @return
*/
public User getUser2() {
User user = new User(1L, "莊子", "123456");
//發(fā)布事件
MyEvent event = new MyEvent(this, user);
applicationContext.publishEvent(event);
return user;
}
}在 service 中注入 ApplicationContext,在業(yè)務(wù)代碼處理完之后,通過 ApplicationContext 對象手動發(fā)布 MyEvent 事件,這樣我們自定義的監(jiān)聽器就能監(jiān)聽到,然后處理監(jiān)聽器中寫好的業(yè)務(wù)邏輯。
最后,在 Controller 中寫一個接口來測試一下
@Resource
private UserService userService;
@GetMapping("/publish")
public String eventPublish() {
userService.getUser2();
return "success";
}執(zhí)行流程:調(diào)用服務(wù),發(fā)布事件,事件發(fā)布后,會被監(jiān)聽到
瀏覽器訪問 http://localhost:8001/listener/publish

