亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > JSP > 正文

Spring AOP 動態多數據源的實例詳解

2024-09-05 00:18:51
字體:
來源:轉載
供稿:網友

我們在開發項目的時候經常會遇到數據源這個問題,特別是在數據同步的時候,下面錯新技術頻道小編就給大家介紹Spring AOP 動態多數據源的實例詳解。

?Spring AOP 動態多數據源的實例詳解

當項目中使用到讀寫分離的時候,我們就會遇到多數據源的問題。多數據源讓人最頭痛的,不是配置多個數據源,而是如何能靈活動態的切換數據源。例如在一個spring和Mybatis的框架的項目中,我們在spring配置中往往是配置一個dataSource來連接數據庫,然后綁定給sessionFactory,在dao層代碼中再指定sessionFactory來進行數據庫操作。

?

正如上圖所示,每一塊都是指定綁死的,如果是多個數據源,也只能是下圖中那種方式。

可看出在Dao層代碼中寫死了兩個SessionFactory,這樣日后如果再多一個數據源,還要改代碼添加一個SessionFactory,顯然這并不符合開閉原則。

那么正確的做法應該是:

具體代碼與配置如下:

1、applicationContext-mgr.xml

<?xml version="1.0" encoding="utf-8" ?><beans xmlns="http://www.springframework.org/schema/beans"  xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"   xmlns:aop="http://www.springframework.org/schema/aop"  xmlns:context="http://www.springframework.org/schema/context"   xmlns:tx="http://www.springframework.org/schema/tx"  xmlns:p="http://www.springframework.org/schema/p"  xsi:schemaLocation="http://www.springframework.org/schema/beans     http://www.springframework.org/schema/beans/spring-beans-2.5.xsdhttp://www.springframework.org/schema/aop/spring-aop-2.5.xsd    http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-2.5.xsd    http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-2.5.xsd">  <!-- use annotation -->  <context:annotation-config />    <context:component-scan base-package="com.carl.o2o.**.mgr">  </context:component-scan>  <!-- master -->  <bean id="master" class="com.mchange.v2.c3p0.ComboPooledDataSource">    <property name="driverClass" value="${driverClassName_master}"/>    <property name="user" value="${username_master}"/>    <property name="password" value="${password_master}"/>    <property name="jdbcUrl" value="${url_master}?Unicode=true&characterEncoding=UTF-8&allowMultiQueries=true"/>    <property name="maxPoolSize" value="150"/>     <property name="minPoolSize" value="10"/>     <property name="initialPoolSize" value="20"/>     <property name="maxIdleTime" value="3600"/>     <property name="acquireIncrement" value="10"/>     <property name="idleConnectionTestPeriod" value="1800"/>    </bean>  <!-- slave -->  <bean id="slave" class="com.mchange.v2.c3p0.ComboPooledDataSource">    <property name="driverClass" value="${driverClassName_slave}"/>    <property name="user" value="${username_slave}"/>    <property name="password" value="${password_slave}"/>    <property name="jdbcUrl" value="${url_slave}?Unicode=true&characterEncoding=UTF-8"/>    <property name="maxPoolSize" value="150"/>     <property name="minPoolSize" value="10"/>     <property name="initialPoolSize" value="20"/>     <property name="maxIdleTime" value="3600"/>     <property name="acquireIncrement" value="10"/>     <property name="idleConnectionTestPeriod" value="1800"/>    </bean>  <!-- spring 動態數據源 -->  <bean id="dynamicDataSource" class="com.carl.dbUtil.DynamicDataSource">    <property name="targetDataSources">       <map key-type="java.lang.String">         <entry key="slave" value-ref="slave" />       </map>     </property>     <property name="defaultTargetDataSource" ref="master" />     </bean>   <!-- mybatis mapper config -->  <bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean">    <property name="dataSource" ref="dynamicDataSource"/>    <property name="configLocation" value="classpath:o2o_mybatis_config.xml"/>    <property name="mapperLocations" >      <list>        <value>classpath:sqlMap/*.xml</value>        <value>classpath*:/com/carl/o2o/**/*.xml</value>      </list>    </property>  </bean>  <bean id="sqlSession" class="org.mybatis.spring.SqlSessionTemplate">    <constructor-arg index="0" ref="sqlSessionFactory"></constructor-arg>  </bean>  <bean class="org.mybatis.spring.mapper.MapperScannerConfigurer">    <property name="basePackage" value="com.carl.o2o.**.mgr.dao" />  </bean>  <!-- 多數據源 aop -->  <bean id="DataSourceAspect" class="com.carl.dbUtil.DataSourceAspect" />  <aop:config>     <aop:advisor pointcut="execution(* com.carl.o2o.mgr.*.*(..))" advice-ref="DataSourceAspect" />  </aop:config>   <!-- 事務 -->  <bean name="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">      <property name="dataSource" ref="dynamicDataSource"></property>  </bean> </beans>

2、DynamicDataSource

DynamicDataSource使用Spring中的代碼結合AOP實現多數據源切換.

public class DynamicDataSource extends AbstractRoutingDataSource {  public DynamicDataSource() {  }  protected Object determineCurrentLookupKey() {    return DBContextHolder.getDbType();  }  public Logger getParentLogger() {    return null;  }}

3、DBContextHolder

DynamicDataSource的輔助類,用于實際的切換多數據源。

public class DBContextHolder {  private static ThreadLocal<String> contextHolder = new ThreadLocal();  public static String MASTER = "master";  public static String SLAVE = "slave";  public DBContextHolder() {  }  public static String getDbType() {    String db = (String)contextHolder.get();    if(db == null) {      db = MASTER;    }    return db;  }  public static void setDbType(String str) {    contextHolder.set(str);  }  public static void setMaster() {    contextHolder.set(MASTER);  }  public static void setSlave() {    contextHolder.set(SLAVE);  }  public static void clearDBType() {    contextHolder.remove();  }}

4、DataSourceAspect

多數據源AOP切面編程實現。

public class DataSourceAspect implements MethodBeforeAdvice, AfterReturningAdvice, ThrowsAdvice {  private static final Logger log = LogManager.getLogger(DataSourceAspect.class);  public DataSourceAspect() {  }  public void before(Method m, Object[] args, Object target) throws Throwable {    try {      if(m != null) {        if((m.getName().startsWith("list") || m.getName().startsWith("select") || m.getName().startsWith("get")         || m.getName().startsWith("count")) && !m.getName().contains("FromMaster")) {          DBContextHolder.setDbType("slave");        } else {          DBContextHolder.setDbType("master");        }      }    } catch (Exception var5) {      log.error("data source aspect error.", var5);    }  }  public void after(JoinPoint point) {    log.info("clear db type after method.current id {}", new Object[]{Long.valueOf(Thread.currentThread().getId())});    DBContextHolder.clearDBType();  }  public void afterReturning(Object returnValue, Method method, Object[] args, Object target) throws Throwable {  }  public void afterThrowing(Method method, Object[] args, Object target, Exception ex) throws Throwable {    log.info("current db type {} when exception", new Object[]{DBContextHolder.getDbType()});    DBContextHolder.setDbType("master");  }}

通過以上的介紹,相信大家已經知道了Spring AOP 動態多數據源的實例詳解,大家按照上面的方法來操作,就能獲得想要的效果了。

亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
色青青草原桃花久久综合| 最新69国产成人精品视频免费| 国产精品久久久久久超碰| 国产亚洲a∨片在线观看| 亚洲免费视频一区二区| 中文字幕在线视频日韩| 国产精品自产拍在线观| 精品网站999www| 精品久久久久久久久久久久久| 久久韩国免费视频| 久久久精品免费| 国产精品美女视频网站| 欧美日本啪啪无遮挡网站| 91精品国产高清自在线| 亚洲色图美腿丝袜| 欧美激情久久久| 久久777国产线看观看精品| 日日狠狠久久偷偷四色综合免费| 国产精品美女免费视频| 亚洲亚裔videos黑人hd| 亚洲三级 欧美三级| 欧美日韩精品在线观看| 亚洲成av人乱码色午夜| 97香蕉超级碰碰久久免费的优势| 欧洲中文字幕国产精品| 国产成人精品久久| 欧美在线视频在线播放完整版免费观看| 亚洲精品视频网上网址在线观看| 久久精品人人做人人爽| 日韩久久精品成人| 97精品久久久中文字幕免费| 国产成人综合精品| 最近2019年好看中文字幕视频| 日韩av电影国产| 亚洲欧美在线第一页| 九九九热精品免费视频观看网站| 国产噜噜噜噜噜久久久久久久久| 亚洲影院色无极综合| 久久成人亚洲精品| 亚洲精品美女在线| 免费不卡欧美自拍视频| 国产欧美日韩中文字幕| 久久久久久久成人| 欧洲美女免费图片一区| 欧美日韩精品二区| 欧洲成人在线观看| 亚洲女人天堂视频| 91精品久久久久久久久不口人| 国产网站欧美日韩免费精品在线观看| 亚洲成年网站在线观看| 高清在线视频日韩欧美| 国产精品一区二区三区成人| 久久精视频免费在线久久完整在线看| 久久久久久久一| 日韩精品极品在线观看| 国产精品嫩草影院久久久| 日韩电影中文字幕在线| 久久久久久久久国产精品| 久久久久中文字幕2018| 成人夜晚看av| 日本精品一区二区三区在线播放视频| 亚洲精品一区二区在线| 精品久久久久久中文字幕大豆网| 国产精品视频一区国模私拍| 国产乱人伦真实精品视频| 免费不卡欧美自拍视频| 亚洲裸体xxxx| 国产亚洲精品成人av久久ww| 8x拔播拔播x8国产精品| 欧美在线激情视频| 91精品国产九九九久久久亚洲| 国产欧美日韩丝袜精品一区| 国产成人综合精品在线| 亚洲人成毛片在线播放| 亚洲一区二区三区在线视频| 91精品国产高清久久久久久久久| 亚洲男人的天堂网站| 成人免费自拍视频| 欧美成人午夜免费视在线看片| 日韩中文字幕在线视频播放| 亚洲一区中文字幕| 亚洲专区国产精品| 538国产精品一区二区免费视频| 国内揄拍国内精品少妇国语| 国产日韩在线视频| 中文字幕亚洲二区| 91免费看片网站| 日韩av在线电影网| 亚洲欧洲av一区二区| 91成人天堂久久成人| 欧美区二区三区| 久久黄色av网站| 日本精品性网站在线观看| 永久免费看mv网站入口亚洲| 欧美日韩在线视频观看| 欧洲亚洲妇女av| 国产一区二区欧美日韩| 国产日产久久高清欧美一区| 亚洲片av在线| 成人在线免费观看视视频| 欧美激情国产高清| 日韩美女视频中文字幕| 日韩精品中文字幕久久臀| 国语自产精品视频在免费| 国产精品com| 精品久久久久久中文字幕大豆网| 理论片在线不卡免费观看| 久久久久久久久久久久久久久久久久av| 成人性教育视频在线观看| 91在线免费观看网站| 中文字幕久久久av一区| 日韩免费看的电影电视剧大全| 亚洲免费一在线| 亚洲国内高清视频| www.99久久热国产日韩欧美.com| 亚洲国产成人爱av在线播放| 亚洲成色777777女色窝| 色综久久综合桃花网| 亚洲加勒比久久88色综合| 国产亚洲精品久久久久久牛牛| 欧美精品videosex性欧美| 最新的欧美黄色| 亚洲r级在线观看| 欧美日韩在线视频观看| 精品久久久久久久久久久久久久| 日韩免费在线免费观看| 欧美高清在线播放| 91精品国产高清久久久久久91| 深夜福利亚洲导航| 欧美日韩裸体免费视频| 国产精品久久久久久久久久ktv| 久热国产精品视频| 久久久午夜视频| 92版电视剧仙鹤神针在线观看| 亚洲精品资源在线| 久久久久久久一区二区三区| 成人久久精品视频| 国产亚洲精品久久久久久| 久久人人爽国产| 久久国产精品亚洲| 国产精品色午夜在线观看| 91免费综合在线| 亚洲男人的天堂在线| 欧美日韩国产一中文字不卡| 91久久国产婷婷一区二区| 在线播放日韩欧美| 国产不卡av在线| 日韩第一页在线| 国产xxx69麻豆国语对白| 色偷偷av一区二区三区| 日本精品视频在线观看| 欧美性xxxxxxxxx| 91经典在线视频| 久久亚洲私人国产精品va| 美女扒开尿口让男人操亚洲视频网站| 亚洲第一网中文字幕| 91情侣偷在线精品国产| 亚洲国产成人91精品| 日韩69视频在线观看| 欧美日韩免费观看中文| 国产日韩av高清| 久久影院中文字幕| 亚洲一区av在线播放| 欧美成人中文字幕在线|