偷偷摘套内射激情视频,久久精品99国产国产精,中文字幕无线乱码人妻,中文在线中文a,性爽19p

關(guān)于如何設(shè)計(jì)一個(gè)基于事件驅(qū)動(dòng)架構(gòu)的思考

開(kāi)發(fā) 架構(gòu)
最近一直在思考一個(gè)問(wèn)題:有沒(méi)有這樣一種可能,就是一個(gè)領(lǐng)域模型的狀態(tài)不依賴于外部,它只負(fù)責(zé)接收外部的事件,然后根據(jù)這些事件做出響應(yīng)……

最近一直在思考一個(gè)問(wèn)題:有沒(méi)有這樣一種可能,就是一個(gè)領(lǐng)域模型的狀態(tài)不依賴于外部,它只負(fù)責(zé)接收外部的事件,然后根據(jù)這些事件做出響應(yīng);響應(yīng)分兩種:

1)根據(jù)模型當(dāng)前的內(nèi)存狀態(tài)進(jìn)行業(yè)務(wù)邏輯處理,然后產(chǎn)生事件,注意:這個(gè)過(guò)程不會(huì)改變模型當(dāng)前的內(nèi)存狀態(tài);

2)根據(jù)事件改變自己的狀態(tài);

另外,也是最重要的,領(lǐng)域模型不用關(guān)心自己所產(chǎn)生的事件到底怎么樣了,比如不關(guān)心有沒(méi)有持久化,不關(guān)心是否和別的事件有并發(fā)沖突。它只管根據(jù)自己當(dāng)前的內(nèi)存狀態(tài)做上面這兩點(diǎn)的響應(yīng);

如果這樣的設(shè)想有可能,那領(lǐng)域模型就是真正的中央業(yè)務(wù)邏輯處理器了,和CPU很類(lèi)似了。這樣它才能真正快起來(lái)。

簡(jiǎn)單的說(shuō)就是:事件->模型->事件

模型只管響應(yīng)事件,然后產(chǎn)生新的事件

領(lǐng)域模型就是一黑盒,它只能幫你處理業(yè)務(wù)邏輯,其他的什么處理結(jié)果它一概不關(guān)心;當(dāng)然,領(lǐng)域模型肯定有它自己的狀態(tài),但這個(gè)狀態(tài)是駐留在內(nèi)存的,和領(lǐng)域模型是一體的。

我為什么會(huì)有這個(gè)想法是因?yàn)椋以谙?,為什么要讓領(lǐng)域模型的處理邏輯依賴于它的處理結(jié)果是否被正確順利持久化了?感覺(jué)這很荒唐。

既然領(lǐng)域模型有自己的內(nèi)存狀態(tài)空間,他的所有邏輯也應(yīng)該只依賴于這個(gè)狀態(tài)空間,不再依賴于其他任何外部的東西。

當(dāng)然,以前我們?cè)O(shè)計(jì)的IRepository,實(shí)際背后都是直接從數(shù)據(jù)庫(kù)取。這樣的話,領(lǐng)域模型的狀態(tài)空間就是數(shù)據(jù)庫(kù)了。但是這樣其實(shí)很不好,因?yàn)闉槭裁床挥脙?nèi)存作為領(lǐng)域模型的狀態(tài)空間呢?

現(xiàn)在再想想LMAX就是我剛才的想法的一個(gè)實(shí)際例子。

事件->模型->事件,這樣的設(shè)計(jì),理論上并不需要必須要求單線程來(lái)訪問(wèn)模型,因?yàn)轭I(lǐng)域模型不依賴于任何外部的狀態(tài),只依賴于自己所在存活內(nèi)存空間;單線程有一個(gè)很大的好處就是可以防止并發(fā)沖突的產(chǎn)生。我們其實(shí)完全支持多線程或集群的方式,只不過(guò)這樣會(huì)有可能訪問(wèn)到的領(lǐng)域?qū)ο蟮臓顟B(tài)是了老的,因?yàn)椴煌臋C(jī)器之間的領(lǐng)域模型內(nèi)存對(duì)象的狀態(tài)需要做一些同步,訪問(wèn)到老數(shù)據(jù)的可能性的大小取決于并發(fā)的大小以及機(jī)器之間數(shù)據(jù)同步的快慢;

LMAX之所以用單線程,是考慮了,這單線程的領(lǐng)域模型和性能之間,性能已經(jīng)可以達(dá)到他們的要求了。

這樣的架構(gòu),領(lǐng)域模型中的任何一個(gè)對(duì)象的一次狀態(tài)更新至少會(huì)響應(yīng)兩個(gè)事件,舉個(gè)例子:

1)先響應(yīng)ChangeNoteCommand(command也是一種事件,可以理解為NoteChangeRequested),然后Note模型產(chǎn)生一個(gè)NoteChanged事件;

2)然后該事件(NoteChanged)最終又被發(fā)送到領(lǐng)域模型讓其響應(yīng),此時(shí),領(lǐng)域模型才去更改自己的Note狀態(tài);

經(jīng)過(guò)這兩個(gè)事件的響應(yīng),才完成了Note的最終狀態(tài)的修改;而我們以前都是從數(shù)據(jù)庫(kù)取Note,然后更改,然后保存到數(shù)據(jù)庫(kù)。這樣不慢才怪!

通過(guò)上面的兩次事件響應(yīng),可以換來(lái)領(lǐng)域模型***的吞吐量。剩下的我們只要考慮:消息的序列化和反序列化、消息傳遞的速度、事件持久化的速度、并發(fā)沖突后重試的設(shè)計(jì),以及消息丟失,等問(wèn)題。但這些都不是領(lǐng)域模型該考慮的問(wèn)題。這些外圍的任何問(wèn)題,都不要讓領(lǐng)域模型自己去考慮,我們應(yīng)該對(duì)出現(xiàn)的各種問(wèn)題逐個(gè)尋求解決方案。

每個(gè)問(wèn)題的解決方案我大概理了下我的對(duì)策:

  1. 消息的序列化和反序列化:這個(gè)簡(jiǎn)單,用BinaryFormatter,或更快的開(kāi)源序列化組件,可以達(dá)到每秒10W次每秒;
  2. 消息傳遞的速度:用MSMQ,如果嫌太慢,就用ZeroMq,可以達(dá)到30W消息每秒;
  3. 事件持久化的速度:由于事件都是跟著單個(gè)聚合根,所以我們只要確保單個(gè)聚合根的事件不會(huì)沖突(即沒(méi)有相同的版本號(hào)的事件);為了更快的持久化,我們可以對(duì)事件按照聚合根或者其他方式進(jìn)行分區(qū)存放,不同的服務(wù)器存放不同的聚合根;這樣通過(guò)集群持久化的方式可以實(shí)現(xiàn)多事件同時(shí)被持久化,從而提高整體的事件持久化吞吐量;如單個(gè)mongodb server每秒持久化5000個(gè),那10個(gè)mongodb server能每秒持久化5W個(gè);
  4. 并發(fā)沖突后怎么辦:一般來(lái)說(shuō)就是選擇重試,但為了確保不會(huì)出現(xiàn)不可控的局面(可能由于某種原因一直在重試),那需要設(shè)置一個(gè)***的重試次數(shù);超過(guò)***重試次數(shù)后不再重試,然后記錄日志,以供以后查找問(wèn)題;這里的重試的意思是:重新找到對(duì)應(yīng)該事件的command,然后再次發(fā)送該command給領(lǐng)域模型處理;
  5. 消息丟失:丟失就丟失了唄,呵呵;要是你覺(jué)得消息決不能丟失,那就用可靠的帶持久化功能的消息傳輸隊(duì)列,如MSMQ;當(dāng)然,就算消息丟失了,我們很多時(shí)候都要想想有沒(méi)有影響的,一般來(lái)說(shuō),消息丟失,至少我們是知道程序有問(wèn)題了的,因?yàn)槟P偷臓顟B(tài)此時(shí)一定是不對(duì)的。我們可以通過(guò)在消息發(fā)出時(shí)和接收時(shí)記錄日志,這樣方便以后查找消息是在哪個(gè)環(huán)節(jié)丟的;
  6. 任何其他的異常出現(xiàn),這個(gè)我覺(jué)得如果都是托管代碼,那可以在必要的地方加try catch,然后記錄日志。至于是否要重試,還要看情形;

另外,如果是多線程訪問(wèn)模型,或集群訪問(wèn),那很多時(shí)候訪問(wèn)到的內(nèi)存的領(lǐng)域?qū)ο蟮臓顟B(tài)都是老的,那怎么辦?其實(shí)這不是問(wèn)題,因?yàn)槭录志没臅r(shí)候會(huì)被檢測(cè)到這種并發(fā)重復(fù),然后對(duì)應(yīng)的command會(huì)被重試。

另外,這種架構(gòu),傳輸?shù)氖鞘录?,事件都是很小的,所以不用?dān)心消息傳輸?shù)男阅堋?/p>

目前就想到這些。后續(xù)再完善思路,

***,我一直認(rèn)為:知識(shí)決定命運(yùn),學(xué)習(xí)積累知識(shí),而正確的思維方式是一切高效學(xué)習(xí)的基礎(chǔ)。所以要學(xué)會(huì)如何清晰地思考!

所以,我們最重要的是要學(xué)會(huì)如何思考。

呵呵!

原文鏈接:http://www.cnblogs.com/netfocus/archive/2013/03/26/2982152.html

責(zé)任編輯:林師授 來(lái)源: 博客園
相關(guān)推薦

2024-08-27 12:49:20

2024-04-24 10:38:22

2024-05-13 08:40:02

Go事件驅(qū)動(dòng)編程

2012-12-17 10:50:27

程序員

2012-11-12 10:46:37

2025-05-27 10:15:00

Go開(kāi)發(fā)軟件架構(gòu)

2020-11-11 09:49:12

計(jì)算架構(gòu)

2021-05-20 13:22:31

架構(gòu)運(yùn)維技術(shù)

2023-12-13 10:44:57

事件驅(qū)動(dòng)事件溯源架構(gòu)

2022-11-08 08:35:53

架構(gòu)微服務(wù)移動(dòng)

2023-08-08 08:00:00

架構(gòu)Kafka

2018-11-22 14:09:45

iOS架構(gòu)組件開(kāi)發(fā)

2025-01-22 08:00:00

架構(gòu)秒殺系統(tǒng)Java

2025-10-28 02:00:00

秒殺系統(tǒng)客戶端并發(fā)

2022-06-02 10:35:20

架構(gòu)驅(qū)動(dòng)

2023-07-12 08:30:52

服務(wù)架構(gòu)事件驅(qū)動(dòng)架構(gòu)

2013-07-01 11:01:22

API設(shè)計(jì)API

2021-09-01 08:58:15

項(xiàng)目 UTFailed

2021-12-24 10:59:37

Kubernetes架構(gòu)代碼

2018-09-06 22:49:31

分布式架構(gòu)服務(wù)器
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)