各位肯定都聽過這樣一句話 : "好的架構(gòu)不是設(shè)計(jì)出來的,而是演進(jìn)出來的,沒有完美的架構(gòu),只有不斷演變、不斷完善的架構(gòu)。" 今天我們來看一下1 號(hào)店 App 服務(wù)端架構(gòu)改造的例子,來具體說明架構(gòu)的演變過程,讓你能更深入地理解架構(gòu)演變背后的原因。
2012 年,隨著智能設(shè)備的普及和移動(dòng)互聯(lián)網(wǎng)的發(fā)展,移動(dòng)端逐漸成為用戶的新入口,各個(gè)電商平臺(tái)都開始聚焦移動(dòng)端 App。這個(gè)時(shí)候,1 號(hào)店也開始試水移動(dòng)端購(gòu)物,從那時(shí)起,1 號(hào)店 App 的服務(wù)端架構(gòu)一共經(jīng)歷了三個(gè)版本的變化。接下來,我就為你具體介紹 App 服務(wù)端架構(gòu)變化的過程以及原因。
V1.0 架構(gòu)
我先說說最開始的 1.0 版本。當(dāng)時(shí)的情況是,App 前端的 IOS 和 Android 開發(fā)團(tuán)隊(duì)是外包出去的,而 App 的服務(wù)端是由 1 號(hào)店內(nèi)部一個(gè)小型的移動(dòng)團(tuán)隊(duì)負(fù)責(zé)的,這個(gè)團(tuán)隊(duì)主要負(fù)責(zé)提供 App 前端需要的各個(gè)接口,接口使用的通信協(xié)議是 HTTP+JSON。具體的架構(gòu)如下圖所示:
image-20220529213018561
這個(gè)架構(gòu)比較簡(jiǎn)單,App 的服務(wù)端整體上就一個(gè)應(yīng)用,由移動(dòng)團(tuán)隊(duì)來維護(hù)所有對(duì)外接口,服務(wù)端內(nèi)部有很多 Jar 包,比如商品搜索、商品詳情、購(gòu)物車等等,這些 Jar 包包含了各個(gè)業(yè)務(wù)線的業(yè)務(wù)邏輯及數(shù)據(jù)庫(kù)訪問,它們由各個(gè)業(yè)務(wù)線的開發(fā)者負(fù)責(zé)提供。
你可以看到,這個(gè) 1.0 版本的服務(wù)端,實(shí)際上就是一個(gè)單體應(yīng)用,只是對(duì)外的接口和內(nèi)部 Jar 包分別由不同的團(tuán)隊(duì)來提供,這個(gè)架構(gòu)的優(yōu)點(diǎn)和缺點(diǎn)同樣都非常明顯。
它的優(yōu)點(diǎn)是簡(jiǎn)單方便。 App 前端的外包團(tuán)隊(duì)只需要對(duì)接后端的一個(gè)移動(dòng)團(tuán)隊(duì)就可以了,然后移動(dòng)團(tuán)隊(duì)通過現(xiàn)成的 Jar 包,封裝各個(gè)業(yè)務(wù)線的功能。至于這些 Jar 包,業(yè)務(wù)線團(tuán)隊(duì)也無(wú)需額外去開發(fā)。
為什么呢?我們知道,早期的電商平臺(tái)都是先有 PC 端應(yīng)用,再推 App,App 最開始的功能,大多是從已有的 PC 端平移過來的。因此,這些 Jar 包直接從 PC 端應(yīng)用里拿過來就可以了,如果 Jar 包版本有更新,由業(yè)務(wù)線團(tuán)隊(duì)直接同步給移動(dòng)團(tuán)隊(duì)即可。
那這個(gè)架構(gòu)設(shè)計(jì)是不是很完美啊?當(dāng)然不是,不知道你發(fā)現(xiàn)了沒有,其實(shí)這里也存在了很多問題。
第一個(gè)問題:移動(dòng)服務(wù)端對(duì) Jar 包的緊密依賴
移動(dòng)團(tuán)隊(duì)負(fù)責(zé)對(duì)外接口,但他們非常依賴業(yè)務(wù)團(tuán)隊(duì)提供的 Jar 包來實(shí)現(xiàn)業(yè)務(wù)邏輯,這是一種物理上的緊耦合依賴關(guān)系。如果業(yè)務(wù)團(tuán)隊(duì)根據(jù) PC 端的需求,修改了應(yīng)用代碼后,Jar 包也會(huì)隨之修改。那么在實(shí)踐中,經(jīng)常會(huì)出現(xiàn)這樣的情況:業(yè)務(wù)團(tuán)隊(duì)很多時(shí)候,要么忘了同步新的 Jar 包給移動(dòng)團(tuán)隊(duì),要么是新的 Jar 包調(diào)整了類的接口,導(dǎo)致了 App 服務(wù)端的功能有問題,或者直接不可用。
第二個(gè)問題:移動(dòng)團(tuán)隊(duì)的職責(zé)過分復(fù)雜
服務(wù)端為 App 提供的是粗粒度接口,而業(yè)務(wù)團(tuán)隊(duì)的 Jar 包提供的是細(xì)粒度的接口。
因此,移動(dòng)團(tuán)隊(duì)在 Jar 包的基礎(chǔ)上,還需要做很多的業(yè)務(wù)邏輯聚合,很多時(shí)候,這些邏輯還跨多個(gè)業(yè)務(wù)線,導(dǎo)致移動(dòng)團(tuán)隊(duì)對(duì)所有業(yè)務(wù)邏輯都要深入了解。相信你也知道,這是很難做到的。
第三個(gè)問題:團(tuán)隊(duì)并行開發(fā)困難
由于移動(dòng)團(tuán)隊(duì)和業(yè)務(wù)團(tuán)隊(duì)是通過物理 Jar 包進(jìn)行集成的,移動(dòng)團(tuán)隊(duì)直接受業(yè)務(wù)團(tuán)隊(duì)的代碼影響,就導(dǎo)致了團(tuán)隊(duì)之間并行開發(fā)困難,一次大的 App 升級(jí)經(jīng)常需要 2~3 個(gè)月的時(shí)間。
而當(dāng)時(shí)的 1 號(hào)店,需要能盡快地推出 App 端,我們所有的做法都是圍繞這個(gè)目的來的,包括把前端團(tuán)隊(duì)外包出去,后端采用單體架構(gòu),移動(dòng)端功能從 PC 端直接移植過來。所以,從當(dāng)時(shí)的情況來說,這種簡(jiǎn)單的服務(wù)端架構(gòu)和團(tuán)隊(duì)合作模式是非常合適的。
而過了一段時(shí)間,當(dāng)移動(dòng)端的功能已經(jīng)初步具備,我們就需要針對(duì)移動(dòng)自身的特點(diǎn)去組織功能,并能夠快速上線這些新功能。那么,這種單體架構(gòu)加物理 Jar 包耦合的方式,就成為 App 進(jìn)一步發(fā)展的瓶頸。接下來,我們就看下系統(tǒng)是如何通過架構(gòu)升級(jí),來解決這個(gè)問題的。
V2.0 架構(gòu)
到了 2013 年,1 號(hào)店 App 服務(wù)端架構(gòu)升級(jí)到了 V2.0。在這個(gè)時(shí)候,1 號(hào)店自己接手了 App 前端的開發(fā)工作,同時(shí),服務(wù)端接口也由各個(gè)業(yè)務(wù)線團(tuán)隊(duì)直接負(fù)責(zé),這樣,App 前端直接對(duì)接多個(gè)后端應(yīng)用提供的 HTTP 接口。
整體架構(gòu)如下圖所示:
對(duì)于各個(gè)業(yè)務(wù)團(tuán)隊(duì)來說,他們現(xiàn)在走向了前臺(tái),每個(gè)團(tuán)隊(duì)負(fù)責(zé)各個(gè)業(yè)務(wù)線的 App 接口。他們一般采取這樣的做法,一方面,他們以 Web 應(yīng)用的方式,為 PC 端瀏覽器提供訪問;另一方面,針對(duì)移動(dòng)端的訪問需求,他們?cè)?Web 應(yīng)用里面,增加了一些 REST 接口,直接供 App 訪問。在這里,移動(dòng)接口和 Web 應(yīng)用在同一個(gè)工程里開發(fā),作為同一個(gè)應(yīng)用進(jìn)行部署和運(yùn)行。
這里你可以看到,這實(shí)際上就是一種分布式的系統(tǒng)架構(gòu),每塊業(yè)務(wù)由不同的團(tuán)隊(duì)負(fù)責(zé),可以很好地支持團(tuán)隊(duì)之間的并行開發(fā);同時(shí),移動(dòng)接口和 PC 端共享底層業(yè)務(wù)邏輯,有助于快速把 PC 端的功能完整地復(fù)制到 App 端。
這樣,通過 V2.0 架構(gòu)的升級(jí),業(yè)務(wù)線團(tuán)隊(duì)的生產(chǎn)力就被完全釋放了,App 的功能也就快速豐富起來了。
但這種方式也帶來了一系列的問題,我們具體說下。
首先是移動(dòng)端和 PC 端互相干擾的問題。
你可以看到,在同一個(gè)業(yè)務(wù)線內(nèi)部,移動(dòng)接口和 Web 應(yīng)用,物理上是綁定在一起的。很多時(shí)候,PC 端的代碼修改會(huì)影響到移動(dòng)接口,而 Web 應(yīng)用的發(fā)布,也會(huì)導(dǎo)致移動(dòng)接口被動(dòng)地被發(fā)布,如果 PC 端出現(xiàn)功能問題,也會(huì)影響到移動(dòng)接口的可用性。反過來也是一樣的,移動(dòng)接口的需求變化,會(huì)影響到 PC 端的功能。
我們知道,當(dāng)移動(dòng)端發(fā)展到了一定程度,它需要和 PC 端有不同的功能和用戶體驗(yàn),但這種緊耦合的方式,導(dǎo)致了相互之間產(chǎn)生很多不必要的干擾,對(duì)系統(tǒng)的功能和穩(wěn)定性都帶來了負(fù)面影響。
其次是重復(fù)開發(fā)的問題。
移動(dòng)接口除了要給 App 端提供業(yè)務(wù)數(shù)據(jù),還需要考慮一系列系統(tǒng)級(jí)的功能,比如說,安全驗(yàn)證、日志記錄、性能監(jiān)控等等,每個(gè)移動(dòng)接口都需要這些通用功能。
那現(xiàn)在,由于 App 前端是和后端直連的,這就意味著,每個(gè)后端系統(tǒng)都需要獨(dú)自去支持這些系統(tǒng)級(jí)的功能,導(dǎo)致了各個(gè)后端系統(tǒng)重復(fù)開發(fā)。一旦這些通用需求發(fā)生了變化,比如說,我們要對(duì)傳輸數(shù)據(jù)進(jìn)行壓縮,那么,所有的后端系統(tǒng)都需要同步調(diào)整,這樣不但工作量很大,而且也給項(xiàng)目管理也帶來了很大的挑戰(zhàn)。
最后是穩(wěn)定性的問題。
在這里,基于這種直連方式,只要一個(gè)后端系統(tǒng)出問題,就會(huì)直接影響到 App 的可用性,使得 App 整體上非常的脆弱。
之所以會(huì)出現(xiàn)以上這些問題,它的根本原因在于,我們?cè)?App 端,直接照搬了 PC 端的做法,沒有針對(duì)移動(dòng)端自身的特點(diǎn),去做架構(gòu)設(shè)計(jì)。
我們知道,當(dāng) App 發(fā)展到一個(gè)成熟階段時(shí),無(wú)論是業(yè)務(wù)功能,還是非業(yè)務(wù)性功能,和 PC 端都是不同的。所以,在架構(gòu)設(shè)計(jì)上,我們必須能夠支持它們各自不同的特點(diǎn),根據(jù)這個(gè)思路,我們的 App 服務(wù)端架構(gòu)也演變到了 V3.0 版本。
V3.0 架構(gòu)
在 V3.0 版本中,服務(wù)端架構(gòu)包含了兩個(gè)大的升級(jí)。
首先,我們對(duì)每個(gè)業(yè)務(wù)線的服務(wù)端進(jìn)行拆分,讓 App 接口和 PC 端接口各自在物理上獨(dú)立,但它們共享核心的業(yè)務(wù)邏輯。
拆分后的架構(gòu)如下圖所示:
這樣拆分的結(jié)果是,原來大的服務(wù)端變成了 3 個(gè)應(yīng)用,包括一個(gè) App 端接口應(yīng)用,一個(gè) PC 端 Web 應(yīng)用,還有一個(gè)核心業(yè)務(wù)邏輯服務(wù),3 個(gè)部分都是獨(dú)立維護(hù)和部署的。
除此之外,架構(gòu)改造還考慮了移動(dòng)端自身的特點(diǎn)。
一方面,每個(gè)移動(dòng)端接口需要調(diào)用對(duì)應(yīng)的后臺(tái)服務(wù),進(jìn)行業(yè)務(wù)邏輯處理,這個(gè)是個(gè)性化的,每個(gè)接口的處理邏輯都不一樣;另一方面,每個(gè)移動(dòng)端接口都需要進(jìn)行系統(tǒng)級(jí)的功能處理,比如前面所說的安全驗(yàn)證、接口監(jiān)控等,這個(gè)是共性的,每個(gè)接口的處理方式都是一樣的。
那么,在架構(gòu)上,我們就需要把共性的系統(tǒng)級(jí)功能進(jìn)行集中處理,把個(gè)性化的業(yè)務(wù)功能進(jìn)行分散處理。
最后,我們結(jié)合服務(wù)端的應(yīng)用拆分,以及對(duì)移動(dòng)接口本身的改造,落地了服務(wù)端 V3.0 架構(gòu)。
如下圖所示:
在這里,App 前端會(huì)通過移動(dòng)網(wǎng)關(guān)來訪問服務(wù)端接口。這里的網(wǎng)關(guān)主要就是負(fù)責(zé)處理通用的系統(tǒng)級(jí)功能,包括通信協(xié)議適配、安全、監(jiān)控、日志等等;網(wǎng)關(guān)處理完之后,會(huì)通過接口路由模塊,轉(zhuǎn)發(fā)請(qǐng)求到內(nèi)部的各個(gè)業(yè)務(wù)服務(wù),比如搜索服務(wù)、詳情頁(yè)服務(wù)、購(gòu)物車服務(wù)等等。
對(duì)于 PC 端瀏覽器來說,它直接訪問對(duì)應(yīng)的 Web 應(yīng)用,如搜索應(yīng)用、詳情頁(yè)應(yīng)用等,然后這些應(yīng)用也是訪問同樣的內(nèi)部服務(wù)。
這里說明下,當(dāng)時(shí)還沒有流行前后端分離,所以 PC 端有對(duì)應(yīng)的 Web 應(yīng)用,同時(shí)負(fù)責(zé)業(yè)務(wù)邏輯和 UI 展現(xiàn)。現(xiàn)在,你已經(jīng)了解了 V3.0 版本的整體架構(gòu)設(shè)計(jì),接下來,我們就深入移動(dòng)網(wǎng)關(guān),去具體了解下它的內(nèi)部實(shí)現(xiàn)機(jī)制。
移動(dòng)網(wǎng)關(guān)的內(nèi)部實(shí)現(xiàn)
在圖中,你可以看到,整個(gè)移動(dòng)網(wǎng)關(guān)分為三層,自上而下分別是通用層、接口路由層、適配層,接下來我們逐一分析。
通用層
首先是通用層,它負(fù)責(zé)所有系統(tǒng)級(jí)功能的處理,比如通訊協(xié)議適配、安全、監(jiān)控、日志等等,這些功能統(tǒng)一由網(wǎng)關(guān)的通用層進(jìn)行預(yù)處理,避免了各個(gè)業(yè)務(wù)線的重復(fù)開發(fā)。
在具體實(shí)現(xiàn)時(shí),每個(gè)通用功能的處理邏輯都會(huì)封裝成一個(gè)攔截器,這些攔截器遵循統(tǒng)一的接口定義,并且攔截器都是可配置的。當(dāng)有外部請(qǐng)求過來,網(wǎng)關(guān)會(huì)依次調(diào)用這些攔截器,完成各個(gè)系統(tǒng)級(jí)功能的處理。
這個(gè)攔截器接口的定義如下:
Object filter(Object input)throws Exception
接口路由層
接下來是接口路由層。移動(dòng)端請(qǐng)求經(jīng)過通用層的預(yù)處理之后,將會(huì)進(jìn)一步分發(fā)給后端的業(yè)務(wù)適配器進(jìn)行處理。
我們?cè)谂渲梦募?,?duì)接口請(qǐng)求的 URL 和業(yè)務(wù)適配器進(jìn)行映射,接口路由層的分發(fā)邏輯就是根據(jù)請(qǐng)求中的 URL,在配置文件里找到對(duì)應(yīng)的適配器,然后把請(qǐng)求交給適配器進(jìn)行后續(xù)的處理。
配置文件的具體內(nèi)容如下所示:
www.website.com/search SearchAdapter
www.website.com/detail DetailAdapter
服務(wù)適配層
最后是服務(wù)適配層。我們知道,外部接口的請(qǐng)求格式,往往和內(nèi)部服務(wù)接口的格式是不一樣的。具體到 1 號(hào)店當(dāng)時(shí)的情況,外部接口是 HTTP+JSON 格式,內(nèi)部服務(wù)是 Hessian+ 二進(jìn)制格式。
適配器首先用來解決內(nèi)外部接口的適配,除此之外,適配器還可以根據(jù)需要,對(duì)多個(gè)內(nèi)部服務(wù)做業(yè)務(wù)聚合,這樣可以對(duì) App 前端提供粗粒度的接口服務(wù),減少遠(yuǎn)程網(wǎng)絡(luò)的調(diào)用次數(shù)。
這些適配器遵循統(tǒng)一的接口定義:
Object adapter(Object input)throws Exception
這些適配器物理上是 Jar 包的形式,由各個(gè)業(yè)務(wù)線研發(fā)團(tuán)隊(duì)提供,所有的適配器會(huì)集中部署在網(wǎng)關(guān),而網(wǎng)關(guān)本身可以支持多實(shí)例的部署,通過水平擴(kuò)展的方式提升服務(wù)端的處理能力。
現(xiàn)在,你已經(jīng)很清楚了 V3.0 架構(gòu)的實(shí)現(xiàn)細(xì)節(jié),接下來,我們就深入看下,這次架構(gòu)升級(jí)達(dá)到了什么樣的實(shí)際效果。
架構(gòu)的實(shí)際效果
首先,App 端和 PC 端徹底獨(dú)立了。在上面的圖中,我們可以看到,App 前端和 PC 端瀏覽器是完全對(duì)等的,PC 端瀏覽器有自己的服務(wù)端,App 前端也有自己的服務(wù)端,在這里,移動(dòng)網(wǎng)關(guān)就充當(dāng) App 服務(wù)端的角色。
在這個(gè)架構(gòu)下,兩個(gè)服務(wù)端都可以針對(duì)自身的特點(diǎn),獨(dú)立開發(fā),獨(dú)立部署,無(wú)論在邏輯層面還是物理層面都實(shí)現(xiàn)了徹底解耦。我們知道,一開始,App 是依附于 PC 端,而現(xiàn)在,它終于可以獨(dú)立地發(fā)展了。
其次,通過架構(gòu)改造,實(shí)現(xiàn)了核心業(yè)務(wù)的復(fù)用。 這里,我們把核心的業(yè)務(wù)邏輯從 Web 應(yīng)用中剝離出來,變成了共享的服務(wù)。在服務(wù)設(shè)計(jì)時(shí),我們不再區(qū)分 PC 端還是移動(dòng)端,而是從業(yè)務(wù)本身出發(fā),提供一套通用的接口,同時(shí)供 PC 端和移動(dòng)端調(diào)用,從而實(shí)現(xiàn)了底層業(yè)務(wù)邏輯的復(fù)用。
還有,這個(gè)架構(gòu)強(qiáng)化了系統(tǒng)級(jí)功能。 原來通用的系統(tǒng)級(jí)功能,由各個(gè)團(tuán)隊(duì)各自去提供,很多團(tuán)隊(duì)要么不提供,要么實(shí)現(xiàn)的方式不一樣;現(xiàn)在的系統(tǒng)級(jí)功能,是由集中式的移動(dòng)網(wǎng)關(guān)統(tǒng)一來提供,我們就可以很方便地強(qiáng)化這些系統(tǒng)級(jí)功能。
舉個(gè)例子,我們可以把通信協(xié)議由 HTTP 升級(jí)為更安全的 HTTPS,當(dāng)后端服務(wù)有問題時(shí),也可以通過網(wǎng)關(guān)進(jìn)行事先的數(shù)據(jù)緩存,直接返回給 App 前端。比如說商品的詳情數(shù)據(jù),就很適合這樣的處理。
所以,有了移動(dòng)網(wǎng)關(guān),整個(gè) App 的可用性、穩(wěn)定性和安全性都得到了大幅度的提升。
最后,團(tuán)隊(duì)分工也更明確了。 在這里,移動(dòng)團(tuán)隊(duì)主要負(fù)責(zé)移動(dòng)網(wǎng)關(guān),包括網(wǎng)關(guān)本身和各種過濾器的維護(hù),他們可以針對(duì)移動(dòng)端的特點(diǎn),做各種系統(tǒng)級(jí)功能的優(yōu)化;而業(yè)務(wù)團(tuán)隊(duì),主要負(fù)責(zé)各自的業(yè)務(wù)邏輯,包括適配器和底層服務(wù)。移動(dòng)團(tuán)隊(duì)和業(yè)務(wù)團(tuán)隊(duì)通過明確的適配接口進(jìn)行協(xié)作,相互不影響。
我們可以看到,V3.0 在 V2.0 分布式架構(gòu)的基礎(chǔ)上,通過服務(wù)化改造,實(shí)現(xiàn)了基礎(chǔ)業(yè)務(wù)的復(fù)用;同時(shí),通過移動(dòng)網(wǎng)關(guān)落地系統(tǒng)級(jí)功能,實(shí)現(xiàn)了系統(tǒng)的平臺(tái)化改造。
總的改造結(jié)果就是,解放了業(yè)務(wù)線,提升了系統(tǒng)的穩(wěn)定性,使得移動(dòng)端可以做大做強(qiáng)。
總結(jié)
今天,我與你分享了 1 號(hào)店 App 服務(wù)端架構(gòu)改造的實(shí)際例子。在這個(gè)例子中,架構(gòu)經(jīng)歷了單體架構(gòu)到分布式架構(gòu),再到 SOA 架構(gòu)的變化過程,并且通過移動(dòng)網(wǎng)關(guān)的方式,一定程度上實(shí)現(xiàn)了平臺(tái)化。在這里,你可以清晰地看到,公司每個(gè)階段的業(yè)務(wù),都有它不同的特點(diǎn),我們選擇的架構(gòu)必須能夠適配它,過度設(shè)計(jì)和設(shè)計(jì)不足,同樣都是有害的。
通過今天的分享,相信你對(duì)各種架構(gòu)的優(yōu)缺點(diǎn),以及業(yè)務(wù)上的適用性有了更進(jìn)一步的了解。他山之石,可以攻玉。架構(gòu)的策略和原則是通用的,希望你能夠通過實(shí)戰(zhàn)不斷去領(lǐng)會(huì)和運(yùn)用。
來源:公眾號(hào)——JAVA日知錄