摘要:題記真的猛士,敢于不做設計,直接開始編碼面對業(yè)務系統(tǒng)中最復雜的部分狀態(tài)模型,有多少程序員就有多少種實現(xiàn)。聊聊狀態(tài)模型設計上常遇到的問題和解決的思路吧。
題記
真的猛士,敢于不做設計,直接開始編碼——面對業(yè)務系統(tǒng)中最復雜的部分:狀態(tài)模型,有多少程序員就有多少種實現(xiàn)。聊聊狀態(tài)模型設計上常遇到的問題和解決的思路吧。
正文有時候我們想做一個富含業(yè)務行為,而又足夠通用的技術架構時,剛開始都是信心滿滿,采用各種設計方法,充分考慮未來的需求,畫出系統(tǒng)依賴、數(shù)據(jù)模型甚至核心類圖,上線時各種性能爆表或者擴展輕松;上線半年之后畫風一轉(zhuǎn),代碼堆得到處都是,哪怕再小心維護依然無法逃離“一年一重構”的魔咒,兩年過后連測試同學的TC都寫不出來,發(fā)生了什么?
寫到這里又到晚上了,不湊巧零食都被清理干凈,餓得天昏地暗(⊙o⊙)…就設想一下這樣一個場景吧(下面的討論只做場景討論,并非真實業(yè)務系統(tǒng)的設計,請專業(yè)的同行們不要見怪。
一分鐘速成方案接到一個炒菜機器人的項目,要求能夠按照吃貨的設計做出各種菜式
大家應該比較熟悉需求或者領域驅(qū)動的套路吧,抄起自上而下設計的錘子開始敲釘子:首先,我們的平臺中會有
廚具:各種廚具的基本使用接口和參數(shù)規(guī)范
菜譜:操作指導
先在腦海中預演一下這樣的設計是怎么運作的:
首先系統(tǒng)應該能夠認識各種不同的廚具,并且知道如何操作它們
案板:切菜程序
炒鍋:翻炒程序、油炸程序
燉鍋:水煮程序、焯水程序
攪拌機:攪拌程序
……
以及它們的清洗程序沒有一一列出
接下來是菜譜,就先來個番茄炒蛋吧
打蛋流程:使用攪拌機,調(diào)整參數(shù)使其能夠打出均勻的蛋液
番茄流程:使用案板,調(diào)整參數(shù)使其能夠切出合適的番茄塊
炒鍋流程:使用炒鍋,先放油,燒熱,放蛋,翻炒,放番茄,翻炒,加鹽
出鍋流程:使用盤子,出鍋
畢竟我們花了1分鐘設計出來的廚具+菜譜架構,看看再做幾個需求會變成什么樣子,需求方要求在能做番茄炒蛋的基礎上,做個辣椒絲炒蛋:
打蛋流程:使用攪拌機,調(diào)整參數(shù)使其能夠打出均勻的蛋液
辣椒流程:使用案板,調(diào)整參數(shù)使其能夠切出合適的辣椒絲
炒鍋流程:使用炒鍋,先放油,燒熱,放蛋,翻炒,放辣椒,翻炒,加鹽
出鍋流程:使用盤子,出鍋
做到這里,一些同學指出,多數(shù)雞蛋搭配的菜譜都是擁有四個標準流程節(jié)點:打蛋、切菜、翻炒、裝盤,而在切菜環(huán)節(jié)中,我們只需要調(diào)整參數(shù)類型和數(shù)值,就可以搭配出“*炒雞蛋”的菜色,至于翻炒環(huán)節(jié)相對麻煩一些,需要加入很多細粒度的操作才能做出適用于業(yè)務發(fā)展的擴展性來;接下來的工作重點,要放在翻炒流程的設計上,開放出盡可能多的SPI,讓第三方在我們這個平臺上共同實現(xiàn)翻炒市場。
系統(tǒng)上線半年,出現(xiàn)了各種業(yè)務分支:不僅原先官方提供的的番茄炒蛋和辣椒炒蛋獲得很好的市場反饋,微調(diào)參數(shù)就輕松支持了苦瓜炒蛋、木耳炒蛋甚至榴蓮炒蛋;業(yè)務方出現(xiàn)了:我們要開辟湯類市場,先從番茄蛋湯開始吧:
打蛋流程:使用攪拌機,調(diào)整參數(shù)使其能夠打出均勻的蛋液
切菜流程:使用案板,調(diào)整參數(shù)切出番茄丁和香蔥段
湯鍋流程:使用湯鍋,加水,燒熱,放蛋,加熱,放番茄,放蔥段,加鹽
出鍋流程:使用湯碗,出鍋
針對原來設計的四套流程,在切菜流程中加入了類似炒鍋流程的多操作支持,接下來又實現(xiàn)了一套全新的湯鍋流程,出鍋也做了些定制。
腦補一下接下來的紅燒肘子、魚香肉絲、清燉羊肉、回鍋肉該怎么實現(xiàn)吧(唉,快餓死了,話說好多程序員做飯都是好手,是真的吧?)
回過頭看看之前的實現(xiàn),核心流程節(jié)點不一定只有4個,每個流程節(jié)點下面的子節(jié)點可能有多個,如果現(xiàn)在要針對業(yè)務方提出的這些葷菜做個重構,該怎么做?
將菜譜系統(tǒng)做成一個多維數(shù)組,就像這個樣子:
菜譜ID[子流程ID],然后分別實現(xiàn)這些流程節(jié)點并將它們存儲在這個菜譜表格中
看起來應該比較完善了吧,可程序員的第六感還是隱隱約約覺得有哪里不對勁,譬如,在番茄炒蛋和辣椒絲炒蛋這兩個大體相同的流程中,“翻炒”、“加鹽”這兩個節(jié)點真的是可復用的嗎?
沒錯,現(xiàn)實架構中往往沒這么簡單,因為
番茄炒蛋的湯多,鹽可以在最后加,也可以在打蛋的時候加,而辣椒炒蛋沒什么湯,鹽要在打蛋的時候加進去
好吧,在不影響流程系統(tǒng)的情況下,我們硬著頭皮在打蛋和炒鍋流程節(jié)點上加了個IF判斷(is 番茄炒蛋),然后就……中招了。
從邏輯來講,這一個小小的IF,將我們原先設計的三維數(shù)組變了個味,把流程圖畫出來可能是這樣的:
IF...ELSE分支就像小說里邊的二向箔,看起來像是沒有改變原有系統(tǒng)的邏輯,但可是可但是它可是混雜在源碼中而不是存在于配置中的邏輯,慢慢的,這種隨意的維護和簡單實現(xiàn),開始模糊系統(tǒng)中的主子流程的邊界,接著模糊菜譜和主流程的邊界,將一個有層次的設計一步一步的煮成一鍋皮蛋瘦肉東北亂燉粥。寫代碼的時候很爽,做維護的時候罵娘
這是一個小小的開始,我們可以抱著取舍的心態(tài)說:我們可以通過編碼規(guī)范的方式要求開發(fā)人員在涉及主流程節(jié)點的邏輯上不允許使用IF分支來保護架構,只有細枝末節(jié)的流程可以使用不規(guī)范的編碼方式。
可現(xiàn)實往往沒有那么簡單,流程節(jié)點之間也不是完全沒有上下文依賴的情況,絕大多數(shù)采用狀態(tài)機架構的系統(tǒng)是不會用多維數(shù)組劃分狀態(tài)的(不信你可以去review代碼),通過邏輯分支搭建的橋梁,整個系統(tǒng)變成一個巨大的狀態(tài)機,那么一個高維的狀態(tài)機系統(tǒng)投影到單維的系統(tǒng)中會發(fā)生什么?(很多視頻網(wǎng)站上有個很好的教學系列《Dimensions》,有一部分內(nèi)容關于如何通過球極投影理解四維空間)狀態(tài)機爆炸了,囧
這種實現(xiàn)的問題還不止于此,由于細粒度的流程是建立在廚具的維度上,而每種廚具對付不同的食材時,還是需要做很多定制化的工作,譬如:
打蛋器/打蛋碗是否能夠加鹽
鍋里倒油/倒水/倒醬油
菜板切圈/切片/切絲
就拿菜板舉例,胡蘿卜切片的手法,和包心菜切片的手法必然是不同的;習慣上對待這種問題的解決手段通常有兩種,一種是把切片的代碼放在菜板上實現(xiàn),另一種是將復雜性下沉到各種食材上分別實現(xiàn)
如果在菜板上實現(xiàn),我們就將獲得一個能夠加工天下食材的“超級菜板”,要么是個上帝類,要么是個錯綜復雜的巨型Service;
如果在食材上實現(xiàn),為了能讓菜板接受各種不同類型食材作為輸入?yún)?shù),我們很可能會在各種食材的上層抽象一個BaseEdible的基類好傳遞參數(shù),然后要么在菜板上做switch邏輯,要么在BaseEdible中提供各種加工方法的實現(xiàn),譬如切片/切絲,但粉條或者大米怎么切片?它和胡蘿卜除了都能吃以外還有什么共性?
也許有細心的同學開始考慮在食材上用Command方式來實現(xiàn)行為,這可能也是一種很糾結的做法,Debug成本暫且不說,一個幾乎可以發(fā)送所有命令的菜板加上一堆看起來什么命令都能接受的食材實現(xiàn),怎么保證系統(tǒng)不會讓菜板去把面粉切個絲,也要做不少工作。
一個看上去很美好的設計,在實施的過程中很可能成為下面三者兼?zhèn)涞脑愀鈱崿F(xiàn)
狀態(tài)機爆炸
上帝類
過度繼承/無用代碼
這時候,比程序員先瘋掉的,大概是聽到程序員說“我做了個小改動,你們回歸一下”這句話的測試同學吧?
換個姿勢怎樣才能讓這個系統(tǒng)像親愛的母上大人一樣,什么菜都會做呢?
回顧前面的設計,鍋碗瓢盆作為容器,它們本身其實沒有發(fā)生過任何變化,只是在盛有不同的食材時,樣子看起來有些不同。按照加工的流程設計狀態(tài)機踩了坑,按照容器狀態(tài)設計狀態(tài)機子節(jié)點也踩了坑,那么我們是不是一開始的出發(fā)點就跑偏了?
我們不妨換個角度來思考,《舌尖上的中國》教育我們:食材很重要,那么是不是可以從食材入手來設計這個系統(tǒng)?
剛才的設計都是以廚師的工作狀態(tài)為出發(fā)點做的:我們手上有各種工具,可以通過各種手段來加工食材。但考慮“做菜”這件事情本身,輸入的是食材,輸出的也是食材,真正發(fā)生狀態(tài)轉(zhuǎn)換的元素是工具嗎?顯然不是,雞蛋到蛋液到炒蛋這個過程中,鍋碗瓢盆沒有任何變化,從食材的角度入手,也是一個好玩的嘗試。
雞蛋蛋有很多加工方法,煮蛋,煎蛋,荷包蛋,蛋液
辣椒也有很多加工方法,辣椒絲,辣椒圈,辣椒片
根據(jù)每一種食材進行抽象,會有一個很好的附帶效果:每個狀態(tài)之間的變遷不可逆而且轉(zhuǎn)移條件沒有多態(tài)行為
碗 -> 洗干凈 -> 干凈的碗
雞蛋 -> 敲開,放入干凈的碗并用筷子攪動 -> 蛋液
雞蛋 -> 敲開,放入干凈的碗并用筷子攪動,加鹽,繼續(xù)攪動 -> 咸蛋液
鍋 -> 洗干凈 -> 干凈的鍋
干凈的鍋 -> 放油,加熱 -> 熱油鍋
蛋液 -> 放入熱油鍋,翻炒 -> 炒雞蛋
番茄 -> 切碎 -> 碎番茄
辣椒 -> 切絲 -> 辣椒絲
辣椒 -> 切圈 -> 辣椒圈
荷包蛋很難變回生雞蛋的樣子,對吧?話說還真有人能做到,不過即使能變回來也不影響整體的設計
這樣就有了材料的狀態(tài)機和轉(zhuǎn)移函數(shù),之前的架構中,加工工具變成了Services或者Utils,鍋具仍然維持細粒度狀態(tài)機,但不再是接受各種食材的上帝組件,我們可以在這樣細粒度的狀態(tài)機模型上進行很細致的加工
在不同菜色的加工過程中,這些細粒度的狀態(tài)機節(jié)點和轉(zhuǎn)移函數(shù)其實都是可以完整復用的,列一個番茄炒蛋的上游流程圖,是不是變得清晰一些?
剩下的部分就不劇透了,設計的樂趣不就在這里嗎?真的餓到全身無力扯不動蛋啦!有機會大家直接討論下,因為類似的問題在交易、物流、工單系統(tǒng)中都有大量的實例,偶爾換個思路,收獲沒準不小滴。
炒蛋·交易核心實戰(zhàn)在進入這個章節(jié)之前,我們再回顧一下,大家是否已經(jīng)通過腦補,消滅了炒蛋系統(tǒng)中的那些問題設計?
上帝類
過度繼承
非原子的狀態(tài)機轉(zhuǎn)移函數(shù)
狀態(tài)機爆炸
如果大體上沒什么問題的話,咱們繼續(xù)向交易核心系統(tǒng)的設計上折騰起來!
發(fā)現(xiàn)了嗎,炒蛋和交易核心系統(tǒng)的架構也有許多相似之處,復雜業(yè)務系統(tǒng)編碼中,最困難的工作是:知道在什么情況下做什么事;我們從炒蛋的思路入手,以兩個交易場景看看能不能跑通
按照前文的套路,交易系統(tǒng)的設計同樣也可以例舉出兩種典型的設計路徑:
1、按照資金流、信息流、物流等人們直接感知到的交易元素,自上而下的設計
2、按照商品、資金、優(yōu)惠券、紅包等交易物料,自下而上的設計
很多同學對于前者的思路應該比較熟悉了,我們在這里重點看一看怎樣以后者的路徑進行交易核心的設計,是否能夠找到一種可以適用于更多業(yè)務場景、健壯的交易架構
首先需要明確,交易的本質(zhì)是什么:多個參與者按照約定,進行財物的轉(zhuǎn)移或在參與者之間發(fā)生服務行為
接下來就是如何在系統(tǒng)中體現(xiàn)這些轉(zhuǎn)移或者服務行為了,會計記賬法,在資產(chǎn)核算、資金審計等領域都有廣泛的應用,在我們的交易系統(tǒng)設計中,參照會計手段,對每個交易元素的狀態(tài)進行建模(主要體現(xiàn)在數(shù)據(jù)庫Schema上,本文篇幅所限就不展開介紹會計記賬在訂單存儲上的應用了,以后有機會再敘)
多帶帶抽出狀態(tài)機來看,用會計術語描述,可以把財物轉(zhuǎn)移或服務行為抽象成一個簡單的流程
[A]簽訂
[D]借方已履行
[C]貸方已確認
對于任何一種交易元素,無論是否采用了擔保交易或第三方介入服務的情況(如對于商品而言,從賣家發(fā)貨,經(jīng)過快遞,到達買家的過程中:發(fā)貨及物流進行中的狀態(tài)為D,買家確認收貨為C)當然,完全可以引入更多復雜的細粒度狀態(tài)機。為了敘事簡便,在后面的表格中,我們都用A、D、C三種狀態(tài)來描述交易元素的轉(zhuǎn)移狀態(tài)。
在一個典型的一口價交易流程中,交易流程如下(默認全部都是擔保交易)
下單
買家已付款
賣家已發(fā)貨
買家確認收貨
交易成功
在最樸素的一口價交易流程中,就是買家賣家一手交錢一手交貨的過程,交易元素只有兩個
資金
商品
那么,在交易過程中,這兩種交易元素的狀態(tài)是如何變更的?
類型 | 下單 | 付款 | 發(fā)貨 | 收貨 | 成功 |
---|---|---|---|---|---|
資金 | A | D | D | D | C |
商品 | A | A | D | C | C |
表格中,不難看出交易的各個環(huán)節(jié)中需要進行的操作
下單(錢A->D,引導買家付款)
買家已付款(貨A->D,引導賣家發(fā)貨)
賣家已發(fā)貨(貨D->C,引導買家確認收貨)
買家確認收貨(錢D->C,系統(tǒng)打款給賣家)
交易成功
就這樣,從最細粒度的狀態(tài)機入手,我們獲得了一個能夠直接明確表示每個State和Transition的設計原型
下面我們再找一個更復雜一點的場景入手
某烘焙供應商接入在線交易,由其分銷商以代理的方式引導用戶選購,用戶在分銷商頁面在線選購商品后,付定金,均分給分銷商及供應商,等店鋪備貨完成之后,通知買家付尾款給供應商,然后供應商發(fā)貨,買家收貨,完成交易
首先列出交易流程列表
下單
買家付定金,供應商與分銷商均分
供應商完成備貨
買家付尾款給供應商
供應商發(fā)貨
買家確認收貨
交易成功
再列出交易元素
分銷商傭金
供應商定金
供應商尾款
供應商備貨
供應商商品
列出交易流程狀態(tài)表格
類型 | 下單 | 定金 | 備貨 | 尾款 | 發(fā)貨 | 收貨 | 成功 |
---|---|---|---|---|---|---|---|
傭金 | A | D | D | D | D | D | C |
定金 | A | D | D | D | D | D | C |
備貨 | A | A | C | C | C | C | C |
尾款 | A | A | A | D | D | D | C |
商品 | A | A | A | A | A | D | C |
當然也可以簡化一下
類型 | 下單 | 定金 | 備貨 | 尾款 | 發(fā)貨 | 收貨 | 成功 |
---|---|---|---|---|---|---|---|
傭金 | A | D | - | - | - | - | C |
定金 | A | D | - | - | - | - | C |
備貨 | A | - | DC | - | - | - | - |
尾款 | A | - | - | D | - | - | C |
商品 | A | - | - | - | - | D | C |
如果需要支持使用優(yōu)惠券的交易呢?再加一行就行
類型 | 下單 | 定金 | 備貨 | 尾款 | 發(fā)貨 | 收貨 | 成功 |
---|---|---|---|---|---|---|---|
券(付款減) | A | D | - | - | - | - | C |
券(下單減) | D | - | - | - | - | - | C |
用這樣的細粒度表格,很輕松就可以獲得每個State(下單、付定金),以及State之間的Transitions,表格中,不同的列表示交易環(huán)節(jié),而不同的行,則表示不同的交易元素
至于逆向流程的支持,其實也很簡單,因為表格中已經(jīng)清晰的描述了每種交易元素的狀態(tài),將交易元素的發(fā)起人和接收人互換,進一步區(qū)分交易元素
需要區(qū)分的交易元素主要有
平臺中轉(zhuǎn)或擔保類元素(如現(xiàn)金、紅包、優(yōu)惠券)
不可退換元素(優(yōu)惠券、充值卡)
實物(通常所說)
服務
按照交易元素的逆向特征來設計對應的逆向元素生成策略,就可以不用考慮太多細節(jié),簡便的支持逆向流程
至此,我們獲得了一個通過細粒度狀態(tài)機表示的交易核心模塊,大家也可以再用其他的交易場景試著套用一下,看看有沒有比較好或者不適合的場景
在線上應用的設計上,還要進一步考慮一些其他的工程因素
如何進行狀態(tài)機編碼
如何借助TCC實現(xiàn)最終事務一致性
合約化的交易數(shù)據(jù)庫Schema設計
這些內(nèi)容我們在以后的篇幅中慢慢探討吧 :)
小結在很多復雜業(yè)務系統(tǒng)的設計中,往往因為建模角度的選取造成后續(xù)維護中的困難
在狀態(tài)機/流程引擎的設計上,建議考慮
節(jié)點之間的轉(zhuǎn)移函數(shù)是否多態(tài)?
節(jié)點本身是否多態(tài)?
節(jié)點是否清晰的映射了需求場景中那些真正發(fā)生改變的對象?
新增流程是否需要修改原有代碼?
在類層次設計上,建議考慮
是否存在無用代碼?
是否存在上帝類?
如果存在這些情況,就像蛋糕上的霉斑,看起來只有一星半點,但你敢吃長霉的食物嗎?
附錄 開閉原則還記得“開閉原則”嗎,就一句話:系統(tǒng)(或者理解為系統(tǒng)中的類、模塊、函數(shù))對于“擴展”應該開放,而對于“修改”應該是封閉的。
這里首先需要界定“擴展”的含義:在不改動原有系統(tǒng)代碼的情況下,新增一個類算不算?新增一個方法呢?如果將它們套用“修改”的語義,對于模塊而言新增一個類是修改,對于類而言新增方法也是修改。開閉原則的邊界似乎沒有那么清晰。
我們可以用一個更簡單的思路來界定開閉原則:是否違背了原有的設計初衷
繼承的問題《重構》書中花了很大的篇幅向讀者介紹“代碼的壞味道”,有一個“Unnecessary Code”的說法,大致的意思是繼承體系的基類中存在下游子類不需要的行為,或者必須被糾正的情況,工作個三五年的朋友們都或多或少的遇到過某個類的所有子類都在復寫基類方法的情況吧。
在不少實用主義的架構文章中,都提過“使用組合來代替繼承”的觀點,其中流傳最廣的一個段子(抱歉我也不知道是不是真事)是:James Gosling 的某次演講會后Q&A環(huán)節(jié)中,有人問他,如果重新設計Java語言,你會做什么?JG回答說,我會干掉“類”,倒不是因為“類”本身有問題,而是會用實現(xiàn)(implements)來取代繼承(extends)
就平時項目的經(jīng)歷而言,基類,尤其是業(yè)務系統(tǒng)中的各種基類,是非常難設計的。因為很難在業(yè)務剛開始的時候就預想到它最終(結束維護下線時)的樣子,也常常因為這樣,我們看到的大多數(shù)Base*命名的類時,除了限定參數(shù)類型,它的代碼行為和Object基本無異(譬如BaseCommand,BaseItem,BaseAction我去太多了),讓后續(xù)維護的同事邊罵邊寫代碼。
上帝類又是來自《重構》的點子:“One Class to rule them all, and in the darkness bind them.”,聽起來有點像《魔戒》的臺詞哈,這個理解起來就不像為什么避免繼承那么糾結了,畢竟誰都不愿意維護一段三五千行而且看起來什么事情都能做的代碼吧?哦,有人用一個匯編文件寫了個操作系統(tǒng),咱們的平臺看起來最多需要兩三個類就夠了。
文章版權歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/64817.html
摘要:前端日報精選京東如何配合業(yè)務打造三端融合開發(fā)平臺第期聊聊組件開發(fā)的邊界把握和狀態(tài)驅(qū)動插件拾趣網(wǎng)站開發(fā)微信小程序?qū)崙?zhàn)二全家桶開發(fā)的一個跨三端的應用掘金原生實現(xiàn)五子棋游戲前端學習中文譯年你應該了解的函數(shù)式編程個人文章的設 2017-06-19 前端日報 精選 京東618:如何配合業(yè)務打造JDReact三端融合開發(fā)平臺?【第970期】聊聊vue組件開發(fā)的邊界把握和狀態(tài)驅(qū)動webpack 插件...
摘要:隨著世界各國對云計算的大力投入,以及行業(yè)內(nèi)技術的快速發(fā)展,現(xiàn)階段整個產(chǎn)業(yè)公司均在進行云計算的產(chǎn)業(yè)整合。對于云計算的發(fā)展前景,各大互聯(lián)網(wǎng)巨頭都相當看好,紛紛為此調(diào)整公司的發(fā)展戰(zhàn)略,競相推出自己的云計算產(chǎn)品和服務。隨著世界各國對云計算的大力投入,以及IT行業(yè)內(nèi)技術的快速發(fā)展,現(xiàn)階段整個IT產(chǎn)業(yè)公司均在進行云計算的產(chǎn)業(yè)整合。對于云計算的發(fā)展前景,各大互聯(lián)網(wǎng)巨頭都相當看好,紛紛為此調(diào)整公司的發(fā)展戰(zhàn)略...
摘要:和區(qū)別,前者發(fā)音時,舌頭位置不但不下拉,還要向上顎貼近,那么發(fā)出來音正確后者舌尖向下拉,發(fā)音正確,否則不對。 i:和I區(qū)別,前者發(fā)音時,舌頭位置不但不下拉,還要向上顎貼近,那么發(fā)出來音正確;后者舌尖向下拉,發(fā)音正確,否則不對。
摘要:前端與狀態(tài)現(xiàn)在的前端開發(fā)中,對于狀態(tài)的管理是重中之重。有限狀態(tài)機那么如何更好的管理前端軟件的復雜度的狀態(tài)機思想給出了自己的答案。有限狀態(tài)機并不是一個復雜的概念簡單說,它有三個特征狀態(tài)總數(shù)是有限的。 前提 在現(xiàn)在的前端社區(qū),關于MVVM、Model driven view 之類的概念,已經(jīng)算是非常普及了。React/Vue 這類框架可以算是代表。而自己雖然有 React/Vue 的使用經(jīng)...
摘要:狀態(tài)機模型區(qū)塊鏈用許多的節(jié)點共同模擬了一臺多復本的狀態(tài)機。區(qū)塊鏈共識的四個階段第一階段是加入共識加入共識階段決定了什么樣的節(jié)點可以參與共識協(xié)議。第四階段是退出共識這是常常被忽略的部分。 在接下來的秘猿科技小課堂里,我們會從技術角度、經(jīng)濟模型設計角度、以及共識角度來拆解 Nervos 加密經(jīng)濟網(wǎng)絡中,底層公鏈 CKB 的設計理念。而本文將會作為技術角度核心設計 Cell 模型的預備文章,...
閱讀 2447·2021-11-15 11:36
閱讀 1189·2019-08-30 15:56
閱讀 2252·2019-08-30 15:53
閱讀 1050·2019-08-30 15:44
閱讀 663·2019-08-30 14:13
閱讀 1005·2019-08-30 10:58
閱讀 486·2019-08-29 15:35
閱讀 1307·2019-08-29 13:58