婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av

主頁 > 知識庫 > Spring AOP 動態多數據源的實例詳解

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.xsd
http://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=truecharacterEncoding=UTF-8allowMultiQueries=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=truecharacterEncoding=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 ThreadLocalString> 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 動態多數據源的實例詳解,如有疑問請留言或者到本站社區交流討論,感謝閱讀,希望能幫助到大家,謝謝大家對本站的支持!

您可能感興趣的文章:
  • Spring+Mybatis 實現aop數據庫讀寫分離與多數據庫源配置操作
  • spring aop action中驗證用戶登錄狀態的實例代碼
  • 利用spring AOP記錄用戶操作日志的方法示例
  • 詳解SpringBoot AOP 攔截器(Aspect注解方式)
  • Spring AOP實現Redis緩存數據庫查詢源碼

標簽:內江 大連 黃石 南京 銅川 廣州 樂山 貸款邀約

巨人網絡通訊聲明:本文標題《Spring AOP 動態多數據源的實例詳解》,本文關鍵詞  Spring,AOP,動態,多,數據源,;如發現本文內容存在版權問題,煩請提供相關信息告之我們,我們將及時溝通與處理。本站內容系統采集于網絡,涉及言論、版權與本站無關。
  • 相關文章
  • 下面列出與本文章《Spring AOP 動態多數據源的實例詳解》相關的同類信息!
  • 本頁收集關于Spring AOP 動態多數據源的實例詳解的相關信息資訊供網民參考!
  • 推薦文章
    婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av
    欧美一级日韩不卡播放免费| 精品系列免费在线观看| 久久久影院官网| 日韩一二三四区| 久久久久久免费网| 精品国产乱码91久久久久久网站| 欧美日韩国产一级| 欧美日韩亚洲丝袜制服| 欧美绝品在线观看成人午夜影视| 欧美视频自拍偷拍| 欧美高清一级片在线| 欧美日韩国产系列| 欧美一区二区三区播放老司机| 欧美日韩综合色| 欧美男女性生活在线直播观看| 欧美日韩免费观看一区二区三区| 在线免费av一区| 91国偷自产一区二区三区观看| 欧美三级视频在线| 91精品婷婷国产综合久久性色| 欧美欧美欧美欧美首页| 日韩精品一区二区三区视频 | 精品粉嫩超白一线天av| 欧美不卡一区二区三区| 久久综合视频网| 欧美国产日韩在线观看| 亚洲国产一二三| 日本三级亚洲精品| 国产精品资源在线看| 一本大道久久a久久综合婷婷| 欧美丝袜丝交足nylons图片| 日韩欧美国产综合| 中文字幕av一区二区三区免费看| 一区2区3区在线看| 激情都市一区二区| 欧亚一区二区三区| 日韩欧美中文字幕制服| 国产精品视频yy9299一区| 天天操天天综合网| 成人不卡免费av| 欧美一区二区三区婷婷月色| 国产精品久久久久久久久久久免费看| 亚洲国产sm捆绑调教视频 | 狠狠色狠狠色综合系列| 91麻豆免费视频| 日韩欧美激情在线| 一区二区三区四区国产精品| 精品在线亚洲视频| 日本韩国视频一区二区| 欧美精品一区二区蜜臀亚洲| 亚洲一区中文日韩| 成人动漫一区二区三区| 日韩午夜电影av| 一区二区不卡在线播放| 波多野结衣亚洲一区| 欧美一级片在线| 亚洲国产精品尤物yw在线观看| 国产一本一道久久香蕉| 日韩欧美高清在线| 亚洲自拍与偷拍| 91在线视频免费观看| 国产日韩欧美一区二区三区乱码| 麻豆91在线播放免费| 欧美在线啊v一区| 国产精品每日更新在线播放网址| 久久99最新地址| 欧美一卡2卡三卡4卡5免费| 一区二区三区在线影院| 成人av动漫网站| 国产日韩欧美高清在线| 国产成人亚洲精品青草天美| 亚洲精品在线三区| 韩国一区二区三区| 精品美女在线播放| 国产精品99久久久久久似苏梦涵 | 亚洲chinese男男1069| 色女孩综合影院| 亚洲素人一区二区| 日本高清不卡视频| 亚洲激情av在线| 欧美视频一区二区| 婷婷久久综合九色综合伊人色| 欧美视频一区在线| 免费高清在线视频一区·| 欧美一级在线视频| 免费在线看成人av| 久久免费看少妇高潮| 九九视频精品免费| 国产欧美一区二区三区沐欲| 成人性色生活片| 国产精品久久影院| 色哟哟一区二区三区| 日韩二区三区在线观看| 91精品国产入口在线| 国产精品99久久久久久宅男| 中文字幕制服丝袜成人av| 北条麻妃国产九九精品视频| 亚洲第一成人在线| 日韩免费高清视频| 九九精品一区二区| ...xxx性欧美| 欧美一二三区在线观看| 成人免费三级在线| 亚洲黄色片在线观看| 日韩写真欧美这视频| 不卡免费追剧大全电视剧网站| 亚洲综合丁香婷婷六月香| 26uuu精品一区二区三区四区在线| 国产成人啪免费观看软件 | 久久女同性恋中文字幕| 豆国产96在线|亚洲| 亚洲国产日韩综合久久精品| www国产亚洲精品久久麻豆| 97久久超碰精品国产| 狠狠色丁香久久婷婷综| 一区二区三区高清在线| 精品国产自在久精品国产| 91视频在线观看| 激情成人午夜视频| 亚洲综合免费观看高清完整版在线| 精品日韩成人av| 精品视频123区在线观看| 国产成人av电影在线| 日本va欧美va瓶| 一区二区三区不卡在线观看| 久久嫩草精品久久久精品一| 欧美在线色视频| 国产白丝精品91爽爽久久| 美女视频第一区二区三区免费观看网站| 国产亲近乱来精品视频| 欧美一级日韩免费不卡| 91极品美女在线| 不卡的看片网站| 懂色av一区二区三区蜜臀| 蜜臀a∨国产成人精品| 偷拍一区二区三区| 亚洲午夜激情av| 亚洲欧美日韩综合aⅴ视频| 欧美激情一区二区三区四区| 精品国产乱码久久久久久图片| 欧美老女人在线| 欧美日韩精品一区二区三区四区 | 欧美国产精品v| 日韩欧美一区电影| 欧美日韩色综合| 欧美日韩国产高清一区| 在线观看亚洲成人| 色94色欧美sute亚洲13| 91视频在线看| 91老司机福利 在线| av动漫一区二区| 成人精品一区二区三区中文字幕| 国产综合久久久久久久久久久久| 美女任你摸久久| 国产美女久久久久| 国产一区二区看久久| 国产乱妇无码大片在线观看| 国产sm精品调教视频网站| 成人黄色大片在线观看| 北条麻妃一区二区三区| 99热在这里有精品免费| 一本久久精品一区二区| 色哟哟精品一区| 欧美一级久久久| 2021久久国产精品不只是精品| 中文字幕免费在线观看视频一区| 国产精品色在线观看| 成人免费在线视频观看| 亚洲精品自拍动漫在线| 日本系列欧美系列| 国产一区日韩二区欧美三区| 国产东北露脸精品视频| 成人av资源在线观看| 欧美最新大片在线看| 日韩一区二区三区视频在线| 久久―日本道色综合久久| 亚洲天堂精品在线观看| 亚洲成人av一区二区| 精品亚洲国产成人av制服丝袜 | 亚洲人吸女人奶水| 亚洲国产视频直播| 精品一区二区在线看| 99久久综合色| 欧美一三区三区四区免费在线看| 国产偷国产偷亚洲高清人白洁 | 欧美精品乱码久久久久久按摩| 欧美电影免费观看高清完整版| 国产欧美日本一区二区三区| 亚洲图片欧美一区| 国产精品一区在线观看你懂的| 欧美三日本三级三级在线播放| 欧美www视频| 亚洲午夜电影在线观看| 成人禁用看黄a在线| 欧美一级欧美三级| 亚洲午夜激情网页| 本田岬高潮一区二区三区| 2020日本不卡一区二区视频| 亚洲一区二区三区爽爽爽爽爽| 国产精品99久久久久久似苏梦涵 |