互聯(lián)網(wǎng)廠商在進(jìn)行公有云saas服務(wù)部署的時(shí)候,往往會(huì)面對(duì)多租戶的場(chǎng) 景,多租戶場(chǎng)景的設(shè)計(jì),在架構(gòu)上一般分為二個(gè)層次:
1、計(jì)算集群的多租戶
計(jì)算集群的多租戶顧名思義,就是針對(duì)不同的租戶提供單獨(dú)的服務(wù)集群,不同租戶之間cpu相互獨(dú)立,個(gè)別租戶的流量洪峰不會(huì)影響其它租戶正常服務(wù)。
計(jì)算集群的多租戶實(shí)現(xiàn)有很多種方案,一般做法是,根據(jù)不同租戶辟出獨(dú)立集群,對(duì)外暴露統(tǒng)一的CDN域名或BGP域名,經(jīng)流量調(diào)度到計(jì)算中心內(nèi)部,由Nginx集群或網(wǎng)關(guān)根據(jù)URL中的租戶參數(shù),向租戶所在集群轉(zhuǎn)發(fā)請(qǐng)求。
由于多租戶的場(chǎng)景的普遍性,最近幾年新涌現(xiàn)的新框架和平臺(tái)在設(shè)計(jì)伊始就考慮了多租戶的場(chǎng)景,并集成了相關(guān)技術(shù),比如k8的namespace,結(jié)合calico,可以實(shí)現(xiàn)非常靈活的多租戶計(jì)算分組。
2、數(shù)據(jù)層的多租戶
拿MySQL來(lái)舉例,數(shù)據(jù)層的多租戶一般有三種做法:
1)一個(gè)租戶一個(gè)數(shù)據(jù)庫(kù)
2)一個(gè)租戶一個(gè)schema
3)多個(gè)租戶同一個(gè)schema
其中區(qū)別,可以查詢相關(guān)資料,網(wǎng)上很多,一般Saas服務(wù)從成本等因素考慮,大多數(shù)采用第3種方案,其主要標(biāo)志為在元數(shù)據(jù)設(shè)計(jì)中,有多租戶字段tenant_id。
3、多租戶的數(shù)據(jù)層訪問(wèn)攔截
這里以JAVA+MyBatis來(lái)說(shuō)明。
Mybatis plugin插件支持?jǐn)r截所有提交到Dao層的SQL,并支持對(duì)提交的SQL進(jìn)行改寫(xiě),原理類似于Spring的Interceptor,多租戶Mybatis插件能夠在運(yùn)行時(shí),動(dòng)態(tài)獲取到應(yīng)用上下文中的租戶變量,在執(zhí)行CRUD操作時(shí),自動(dòng)將多租戶字段(tenant_id)條件附加到sql語(yǔ)句之中,如where條件之后。
1)使用mybatis插件的優(yōu)點(diǎn)
使用mybatis插件,可以大大簡(jiǎn)化業(yè)務(wù)代碼在多租戶改造過(guò)程中的開(kāi)發(fā)成本,對(duì)比硬編碼方式,mApper和dao層無(wú)須改造,service層的代碼改造量也可以大大降低。
2)原理
通過(guò)mybatis plugin攔截sql處理步驟和result處理步驟。sql預(yù)處理,自動(dòng)增加tenant_id相關(guān)語(yǔ)句,result預(yù)處理,驗(yàn)證數(shù)據(jù)tenant_id字段符合要求。包括:
- insert into,增加寫(xiě)入tenant_id數(shù)據(jù)
- select 自動(dòng)在where之后附加條件tenant_id = ?
- update 在where之后附加條件tenant_id
- delete 在where之后附加條件tenant_id
- left join
- 子查詢
等等。
3)實(shí)現(xiàn)方式
插件用的是責(zé)任鏈模式,由每一個(gè)對(duì)象對(duì)其下家的引用而連接起來(lái)形成一條鏈,請(qǐng)求在這個(gè)鏈上傳遞,直到鏈上的某一個(gè)對(duì)象決定處理此請(qǐng)求。
具體實(shí)現(xiàn)方式為,實(shí)現(xiàn)接口Interceptor.java 的3個(gè)方法,分別是plugin、intercept和setProperties。
- intercept:它將直接覆蓋你所攔截的對(duì)象,有個(gè)參數(shù)Invocation對(duì)象,通過(guò)該對(duì)象,可以反射調(diào)度原來(lái)對(duì)象的方法;
- plugin:target是被攔截的對(duì)象,它的作用是給被攔截對(duì)象生成一個(gè)代理對(duì)象;
- setProperties:允許在plugin元素中配置所需參數(shù),該方法在插件初始化的時(shí)候會(huì)被調(diào)用一次
實(shí)現(xiàn)類需要添加幾個(gè)注解,來(lái)攔截對(duì)應(yīng)的簽名:
然后實(shí)現(xiàn)Interceptor接口的方法即可,通過(guò)Plugin工具類方便生成代理類,通過(guò)MetaObject工具類方便操作四大對(duì)象的屬性,修改對(duì)應(yīng)的值。
然后實(shí)現(xiàn)Plugin方法,生成代理類的方法是通過(guò)MyBatis提供的Plugin工具類,它實(shí)現(xiàn)了InvocationHandler接口(JDK動(dòng)態(tài)代理的接口),看看它的2個(gè)方法:
Plugin提供了靜態(tài)方法wrap方法,它會(huì)根據(jù)插件的簽名配置,使用JDK動(dòng)態(tài)代理的方法,生成一個(gè)代理類,當(dāng)四大對(duì)象執(zhí)行方法時(shí),會(huì)調(diào)用Plugin的invoke方法,如果方法包含在聲明的簽名里,就會(huì)調(diào)用自定義插件的intercept方法,傳入Invocation對(duì)象。
最后,插件的初始化時(shí)在MyBatis初始化的時(shí)候完成的,讀入插件節(jié)點(diǎn)和配置的參數(shù),使用反射技術(shù)生成插件實(shí)例,然后調(diào)用插件方法中的setProperties方法設(shè)置參數(shù),并將插件實(shí)例保存到配置對(duì)象中.
4、最后分享一下很不錯(cuò)的mybatis多租戶插件,當(dāng)然,你也可以自己實(shí)現(xiàn)
https://github.com/Mearalu/mybatis-multi-tenancy