摘要:最近公司項(xiàng)目需要支持,每個(gè)數(shù)據(jù)庫地址不同,需求就是修改配置文件,添加或者刪除數(shù)據(jù)庫配置,重啟系統(tǒng)后就可以完成,不需要額外修改代碼。那怎么來完成數(shù)據(jù)庫的訪問的呢核心類是,而變量才是完成訪問的關(guān)鍵,而這個(gè)其實(shí)就是。
最近公司項(xiàng)目需要支持multi-tenancy,每個(gè)tenant數(shù)據(jù)庫地址不同,需求就是修改配置文件,添加或者刪除數(shù)據(jù)庫配置,重啟系統(tǒng)后就可以完成,不需要額外修改代碼。
網(wǎng)上搜索了一下,基本上都是將配置寫在了代碼里了,例如:http://www.devzxd.top/2017/06...,不能動(dòng)態(tài)增減數(shù)據(jù)庫配置,并且對(duì)不同的數(shù)據(jù)庫創(chuàng)建不同的Repository。繼續(xù)Google之后,在github上找到一個(gè)例子multi-tenant-spring-mongodb,看了一下代碼,還是不符合需求。沒辦法,只能看代碼,調(diào)試來看看自己能不能實(shí)現(xiàn)這個(gè)需求了。
那spring-data-mongodb怎么來完成數(shù)據(jù)庫的訪問的呢?核心類是SimpleMongoRepository,而mongoOperations變量才是完成訪問的關(guān)鍵,而這個(gè)mongoOperations其實(shí)就是mongoTemplate。那就意味著如果我們能夠動(dòng)態(tài)改變這個(gè)mongoTemplate的值就可以切換數(shù)據(jù)庫了,竊喜。
解決的方法有了,但是如何實(shí)現(xiàn)呢?庫并沒有提供類似Hibernate那樣的MultiTenantConnectionProvider和CurrentTenantIdentifierResolver來幫助我們實(shí)現(xiàn)。既然spring是通過AOP和Proxy來完成功能的調(diào)用的,我們似乎也可以這么玩。
我們要對(duì)所有的Repository方法的訪問創(chuàng)建一個(gè)PointCut,這樣我們就可以在訪問之前搞事情了。
利用反射拿到j(luò)oinPoint的target,然后調(diào)用用AopProxyUtils.getSingletonTarget(target)取到最終的SimpleMongoRepository實(shí)例。
通過反射設(shè)置mongoOperations的值。
代碼如下:
@Repository public interface WidgetDataRepository extends MongoRepository{ }
// 注意:線程不安全!?。?@Around("execution(* com.example.demo.dao.mongo.MongoWidgetDataRepo.*(..))") public Object doSwitch(ProceedingJoinPoint joinPoint) throws Throwable { // 拿到我們需要的tenant String tenant = (String) RequestContextHolder.currentRequestAttributes().getAttribute("tenant", SCOPE_REQUEST); tenant = tenant == null ? "test" : tenant; // 通過反射獲取到target Field methodInvocationField = joinPoint.getClass().getDeclaredField("methodInvocation"); methodInvocationField.setAccessible(true); ReflectiveMethodInvocation o = (ReflectiveMethodInvocation) methodInvocationField.get(joinPoint); Field targetField = o.getClass().getDeclaredField("target"); targetField.setAccessible(true); Object target = targetField.get(o); // 獲得SimpleMongoRepository,并往里面填入值 Object singletonTarget = AopProxyUtils.getSingletonTarget(target); Field mongoOperationsField = singletonTarget.getClass().getDeclaredField("mongoOperations"); mongoOperationsField.setAccessible(true); mongoOperationsField.set(singletonTarget, ac.getBean("mongoTemplate" + tenant)); return joinPoint.proceed(); }
這樣我們就可以完成數(shù)據(jù)庫的切換了,可以利用spring-data-mongodb,盡可能的少些模板代碼。
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/68310.html
摘要:最近公司項(xiàng)目需要支持,每個(gè)數(shù)據(jù)庫地址不同,需求就是修改配置文件,添加或者刪除數(shù)據(jù)庫配置,重啟系統(tǒng)后就可以完成,不需要額外修改代碼。那怎么來完成數(shù)據(jù)庫的訪問的呢核心類是,而變量才是完成訪問的關(guān)鍵,而這個(gè)其實(shí)就是。 最近公司項(xiàng)目需要支持multi-tenancy,每個(gè)tenant數(shù)據(jù)庫地址不同,需求就是修改配置文件,添加或者刪除數(shù)據(jù)庫配置,重啟系統(tǒng)后就可以完成,不需要額外修改代碼。 網(wǎng)上搜...
摘要:踩到許多坑,記錄下一些基于的東西吧首先。先創(chuàng)建個(gè)集合,保存寫信息,然后再創(chuàng)建集合,可以發(fā)現(xiàn)存儲(chǔ)的時(shí)候變成了一個(gè)引用類型。王大錘那么查詢的時(shí)候,如果要根據(jù)查詢班級(jí)怎么辦,的查詢也非常簡(jiǎn)單。詳情可以查看官方文檔用法 剛接觸mongodb不久。踩到許多坑,記錄下一些基于spring-data-mongodb的東西吧 首先。應(yīng)該了解下什么情況下使用mongodb,什么情況下用mysql: 業(yè)...
摘要:踩到許多坑,記錄下一些基于的東西吧首先。王大錘那么查詢的時(shí)候,如果要根據(jù)查詢班級(jí)怎么辦,的查詢也非常簡(jiǎn)單。詳情可以查看官方文檔用法 剛接觸mongodb不久。踩到許多坑,記錄下一些基于spring-data-mongodb的東西吧 首先。應(yīng)該了解下什么情況下使用mongodb,什么情況下用mysql: 業(yè)務(wù)需要事物,使用mysql,因?yàn)閙ongodb不支持事物 數(shù)據(jù)量大,但是數(shù)據(jù)本身...
摘要:踩到許多坑,記錄下一些基于的東西吧首先。王大錘那么查詢的時(shí)候,如果要根據(jù)查詢班級(jí)怎么辦,的查詢也非常簡(jiǎn)單。詳情可以查看官方文檔用法 剛接觸mongodb不久。踩到許多坑,記錄下一些基于spring-data-mongodb的東西吧 首先。應(yīng)該了解下什么情況下使用mongodb,什么情況下用mysql: 業(yè)務(wù)需要事物,使用mysql,因?yàn)閙ongodb不支持事物 數(shù)據(jù)量大,但是數(shù)據(jù)本身...
摘要:版本不支持支持為,如果使用并且使用,則會(huì)出現(xiàn)提示,編譯出錯(cuò)。掃描的倉庫目錄,會(huì)自動(dòng)掃描擴(kuò)展了接口的接口進(jìn)行注入。 mongodb介紹 MongoDB 是一個(gè)基于分布式文件存儲(chǔ)的數(shù)據(jù)庫。由 C++ 語言編寫。旨在為 WEB 應(yīng)用提供可擴(kuò)展的高性能數(shù)據(jù)存儲(chǔ)解決方案。MongoDB 是一個(gè)介于關(guān)系數(shù)據(jù)庫和非關(guān)系數(shù)據(jù)庫之間的產(chǎn)品,是非關(guān)系數(shù)據(jù)庫當(dāng)中功能最豐富,最像關(guān)系數(shù)據(jù)庫的。 MongoD...
閱讀 2454·2019-08-30 15:52
閱讀 2252·2019-08-30 12:51
閱讀 2846·2019-08-29 18:41
閱讀 2829·2019-08-29 17:04
閱讀 826·2019-08-29 15:11
閱讀 1744·2019-08-28 18:02
閱讀 3614·2019-08-26 10:22
閱讀 2519·2019-08-26 10:12