SpringBoot Event 事件同步、異步處理

業(yè)務(wù)需求場景:按照一定的順序做一些事情扼菠,例如向A表插入數(shù)據(jù)事物提交之后摄杂,向B表中插入歷史記錄,最后向C表插入循榆。

事件機制

事件監(jiān)聽機制可以理解為是一種觀察者模式析恢,有數(shù)據(jù)發(fā)布者(事件源)和數(shù)據(jù)接受者(監(jiān)聽器);在Java中秧饮,事件對象都是繼承java.util.EventObject對象映挂,事件監(jiān)聽器都是java.util.EventListener實例;Spring中

Java事件

EventObject

java.util.EventObject是事件狀態(tài)對象的基類,它封裝了事件源對象以及和事件相關(guān)的信息盗尸。所有java的事件類都需要繼承該類柑船。

EventListener

java.util.EventListener是一個標(biāo)記接口,就是說該接口內(nèi)是沒有任何方法的泼各。所有事件監(jiān)聽器都需要實現(xiàn)該接口鞍时。事件監(jiān)聽器注冊在事件源上,當(dāng)事件源的屬性或狀態(tài)改變的時候扣蜻,調(diào)用相應(yīng)監(jiān)聽器內(nèi)的回調(diào)方法寸癌。

Source

事件源不需要實現(xiàn)或繼承任何接口或類,它是事件最初發(fā)生的地方弱贼。因為事件源需要注冊事件監(jiān)聽器蒸苇,所以事件源內(nèi)需要有相應(yīng)的盛放事件監(jiān)聽器的容器。

Spring 事件

在 Spring 中吮旅,初始化容器時會調(diào)用 org.springframework.context.ConfigurableApplicationContext 接口中的 reFresh() 方法進行 Bean的加載溪烤,該方法會進行事件的監(jiān)聽注冊。

代碼實例

UserEvent:user事件對象庇勃,繼承ApplicationEvent

public class UserEvent<T> extends ApplicationEvent {

? ? private T data;

? ? public UserEvent(T source) {

? ? ? ? super(source);

? ? ? ? this.data =source;

? ? }

? ? public T getData() {

? ? ? ? return this.data;

? ? }

? ? public void setData(final T data) {

? ? ? ? this.data = data;

? ? }

}

發(fā)布插入用戶的事件

@Autowired

private ApplicationEventPublisher publisher;

/**

? * 發(fā)布插入user表的事件

? */

private void sendInsertUser() {

? ? User user = new User();

? ? user.setId(20210317);

? ? user.setUsername("測試張三");

? ? UserEvent<User> userEvent = new UserEvent<>(user);

? ? // 發(fā)布事件

? ? publisher.publishEvent(userEvent);

}

插入用戶監(jiān)聽器:同步操作

/**

* 監(jiān)聽插入User的事件

*/

@EventListener

public void insertUserLinster(UserEvent<User> userEvent) {

User data = userEvent.getData();

String message = String.format("get user message: %s", JSON.toJSONString(data));

log.info(message);

// 后續(xù)操作檬嘀;繼續(xù)處理

}

測試代碼:

/**

? * SpringBoot Event 使用

? *

? * @return

? */

@Override

public BackResult testEvent() {

? ? BackResult backResult = new BackResult();

? ? try {

? ? ? ? // 向user 表插入數(shù)據(jù)

? ? ? ? insertUsers();

? ? ? ? // 添加user插入對象操作

? ? ? ? sendInsertUser();

? ? ? ? backResult.setSuccess(true);

? ? ? ? backResult.setMessage("操作成功!");

? ? } catch (Exception e) {

? ? ? ? backResult.setMessage(e.getMessage());

? ? ? ? backResult.setSuccess(false);

? ? }

? ? return backResult;

}

insertUsers:開啟事務(wù)

/**

? * 插入開啟事物

? */

@Transactional(rollbackFor = Exception.class)

public void insertUsers() {

? ? insertUserBatch();

}

思考

測試代碼中,方法執(zhí)行順序應(yīng)該是insertUsers()執(zhí)行完成责嚷,事務(wù)提交之后在執(zhí)行sendInsertUser()鸳兽,所以事件監(jiān)聽器因該改進一下,在事務(wù)提交之后去執(zhí)行罕拂。改進代碼如下揍异。

插入用戶監(jiān)聽器:

/**

? * 監(jiān)聽插入User的事件 : 在上個事務(wù)提交之后在執(zhí)行

? */

@TransactionalEventListener(phase = TransactionPhase.AFTER_COMMIT,fallbackExecution = true)

public void insertUserLinster(UserEvent<User> userEvent) {

User data = userEvent.getData();

String message = String.format("get user message: %s", JSON.toJSONString(data));

log.info(message);

// 后續(xù)操作全陨;繼續(xù)處理

}

還有一個問題就是沒有考慮,同步和異步的問題衷掷,上面的方式是同步的辱姨,這樣會影響正常的業(yè)務(wù)邏輯,為了不影響正常的業(yè)務(wù)操作戚嗅,可以將監(jiān)聽器修改為異步執(zhí)行的即可雨涛。使用 @Async 標(biāo)記即可,注意前提條件是:使用 @EnableAsync 開啟 Spring 異步懦胞。

/**

? * 監(jiān)聽插入User的事件 : 在上個事務(wù)提交之后在執(zhí)行

? */

@Async

@TransactionalEventListener(phase = TransactionPhase.AFTER_COMMIT,fallbackExecution = true)

public void insertUserLinster(UserEvent<User> userEvent) {

? ? User data = userEvent.getData();

? ? String message = String.format("get user message: %s", JSON.toJSONString(data));

? ? log.info(message);

? ? // 后續(xù)操作替久;繼續(xù)處理

}

完整代碼地址 :https://gitee.com/Marlon_Brando/back/commit/de586caed15004d0e8e3ae2e9a59f6f170d726d1

————————————————

版權(quán)聲明:本文為CSDN博主「MarlonBrando1998」的原創(chuàng)文章,遵循CC 4.0 BY-SA版權(quán)協(xié)議躏尉,轉(zhuǎn)載請附上原文出處鏈接及本聲明侣肄。

原文鏈接:https://blog.csdn.net/qq_37248504/article/details/115269995

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市醇份,隨后出現(xiàn)的幾起案子稼锅,更是在濱河造成了極大的恐慌,老刑警劉巖僚纷,帶你破解...
    沈念sama閱讀 216,372評論 6 498
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件矩距,死亡現(xiàn)場離奇詭異,居然都是意外死亡怖竭,警方通過查閱死者的電腦和手機锥债,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,368評論 3 392
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來痊臭,“玉大人哮肚,你說我怎么就攤上這事」愠祝” “怎么了允趟?”我有些...
    開封第一講書人閱讀 162,415評論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長鸦致。 經(jīng)常有香客問我潮剪,道長,這世上最難降的妖魔是什么分唾? 我笑而不...
    開封第一講書人閱讀 58,157評論 1 292
  • 正文 為了忘掉前任抗碰,我火速辦了婚禮,結(jié)果婚禮上绽乔,老公的妹妹穿的比我還像新娘弧蝇。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 67,171評論 6 388
  • 文/花漫 我一把揭開白布看疗。 她就那樣靜靜地躺著沙峻,像睡著了一般。 火紅的嫁衣襯著肌膚如雪鹃觉。 梳的紋絲不亂的頭發(fā)上专酗,一...
    開封第一講書人閱讀 51,125評論 1 297
  • 那天睹逃,我揣著相機與錄音盗扇,去河邊找鬼。 笑死沉填,一個胖子當(dāng)著我的面吹牛疗隶,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播翼闹,決...
    沈念sama閱讀 40,028評論 3 417
  • 文/蒼蘭香墨 我猛地睜開眼斑鼻,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了猎荠?” 一聲冷哼從身側(cè)響起坚弱,我...
    開封第一講書人閱讀 38,887評論 0 274
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎关摇,沒想到半個月后荒叶,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,310評論 1 310
  • 正文 獨居荒郊野嶺守林人離奇死亡输虱,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,533評論 2 332
  • 正文 我和宋清朗相戀三年些楣,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片宪睹。...
    茶點故事閱讀 39,690評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡愁茁,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出亭病,到底是詐尸還是另有隱情鹅很,我是刑警寧澤,帶...
    沈念sama閱讀 35,411評論 5 343
  • 正文 年R本政府宣布罪帖,位于F島的核電站道宅,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏胸蛛。R本人自食惡果不足惜污茵,卻給世界環(huán)境...
    茶點故事閱讀 41,004評論 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望葬项。 院中可真熱鬧泞当,春花似錦、人聲如沸民珍。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,659評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至陋桂,卻和暖如春逆趣,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背嗜历。 一陣腳步聲響...
    開封第一講書人閱讀 32,812評論 1 268
  • 我被黑心中介騙來泰國打工宣渗, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人梨州。 一個月前我還...
    沈念sama閱讀 47,693評論 2 368
  • 正文 我出身青樓痕囱,卻偏偏與公主長得像,于是被迫代替她去往敵國和親暴匠。 傳聞我的和親對象是個殘疾皇子鞍恢,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,577評論 2 353

推薦閱讀更多精彩內(nèi)容