轉自: http://blog.51cto.com/linhongyu/1615895
一、前言
近期一項目A需實現數據同步到另一項目B數據庫中,在不改變B項目的情況下,只好選擇項目A中切換數據源,直接把數據寫入項目B的數據庫中。這種需求,在數據同步與定時任務中經常需要。
那么問題來了,該如何解決多數據源問題呢?不光是要配置多個數據源,還得能靈活動態的切換數據源。以spring+hibernate框架項目為例(引用:http://blog.csdn.net/wangpeng047/article/details/8866239博客的圖片):
單個數據源綁定給sessionFactory,再在Dao層操作,若多個數據源的話,那不是就成了下圖:
可見,sessionFactory都寫死在了Dao層,若我再添加個數據源的話,則又得添加一個sessionFactory。所以比較好的做法應該是下圖:
接下來就為大家講解下如何用spring來整合這些數據源,同樣以spring+hibernate配置為例。
二、實現原理
1、擴展Spring的AbstractRoutingDataSource抽象類(該類充當了DataSource的路由中介, 能有在運行時, 根據某種key值來動態切換到真正的DataSource上。)
從AbstractRoutingDataSource的源碼中:
1 public abstract class AbstractRoutingDataSource extends AbstractDataSource implements InitializingBean
我們可以看到,它繼承了AbstractDataSource,而AbstractDataSource不就是javax.sql.DataSource的子類,So我們可以分析下它的getConnection方法:
獲取連接的方法中,重點是determineTargetDataSource()方法,看源碼:
上面這段源碼的重點在於determineCurrentLookupKey()方法,這是AbstractRoutingDataSource類中的一個抽象方法,而它的返回值是你所要用的數據源dataSource的key值,有了這個key值,resolvedDataSource(這是個map,由配置文件中設置好后存入的)就從中取出對應的DataSource,如果找不到,就用配置默認的數據源。
看完源碼,應該有點啟發了吧,沒錯!你要擴展AbstractRoutingDataSource類,並重寫其中的determineCurrentLookupKey()方法,來實現數據源的切換:
DataSourceHolder這個類則是我們自己封裝的對數據源進行操作的類:
2、有人就要問,那你setDataSource這方法是要在什么時候執行呢?當然是在你需要切換數據源的時候執行啦。手動在代碼中調用寫死嗎?這是多蠢的方法,當然要讓它動態咯。所以我們可以應用spring aop來設置,把配置的數據源類型都設置成為注解標簽,在service層中需要切換數據源的方法上,寫上注解標簽,調用相應方法切換數據源咯(就跟你設置事務一樣):
當然,注解標簽的用法可能很少人用到,但它可是個好東西哦,大大的幫助了我們開發:
1 package com.datasource.test.util.database; 2 3 import java.lang.annotation.*; 4 5 @Target({ElementType.METHOD, ElementType.TYPE}) 6 @Retention(RetentionPolicy.RUNTIME) 7 @Documented 8 public @interface DataSource { 9 String name() default DataSource.master; 10 11 public static String master = "dataSource1"; 12 13 public static String slave1 = "dataSource2"; 14 15 public static String slave2 = "dataSource3"; 16 17 }
三、配置文件
為了精簡篇幅,省略了無關本內容主題的配置。
項目中單獨分離出application-database.xml,關於數據源配置的文件。
四、疑問
多數據源切換是成功了,但牽涉到事務呢?單數據源事務是ok的,但如果多數據源需要同時使用一個事務呢?這個問題有點頭大,網絡上有人提出用atomikos開源項目實現JTA分布式事務處理。你怎么看?
五、dataSourceExchange 是怎樣寫的?
dataSourceExchange對應的類可以實現接口org.aopalliance.intercept.MethodInterceptor的invoke方法|@|@Override|@|public Object invoke(MethodInvocation invocation) throws Throwable {|@| DataSource dataSource = invocation.getMethod().getAnnotation(DataSource.class); |@| DataSourceHolder.setDataSource(dataSource.name());|@| try {|@| invocation.proceed();|@| } catch (Exception ex) { |@| }|@| return null;|@|}|@|pointcut的expression也可以寫成@annotation(com.xxx.DataSource)|@|使用的時候,只需要在方法上加上注解@DataSource就行了|@|@DataSource(name = DataSource.slave1)|@|public void insert(String name) {|@|}