DDR爱好者之家 Design By 杰米

 Spring AOP 动态多数据源的实例详解

当项目中使用到读写分离的时候,我们就会遇到多数据源的问题。多数据源让人最头痛的,不是配置多个数据源,而是如何能灵活动态的切换数据源。例如在一个spring和Mybatis的框架的项目中,我们在spring配置中往往是配置一个dataSource来连接数据库,然后绑定给sessionFactory,在dao层代码中再指定sessionFactory来进行数据库操作。

 Spring AOP 动态多数据源的实例详解

正如上图所示,每一块都是指定绑死的,如果是多个数据源,也只能是下图中那种方式。

Spring AOP 动态多数据源的实例详解

可看出在Dao层代码中写死了两个SessionFactory,这样日后如果再多一个数据源,还要改代码添加一个SessionFactory,显然这并不符合开闭原则。

那么正确的做法应该是:

Spring AOP 动态多数据源的实例详解

具体代码与配置如下:

1、applicationContext-mgr.xml

<"1.0" encoding="utf-8" "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}"/>

    <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}"/>

    <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 动态多数据源的实例详解,如有疑问请留言或者到本站社区交流讨论,感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!

DDR爱好者之家 Design By 杰米
广告合作:本站广告合作请联系QQ:858582 申请时备注:广告合作(否则不回)
免责声明:本站资源来自互联网收集,仅供用于学习和交流,请遵循相关法律法规,本站一切资源不代表本站立场,如有侵权、后门、不妥请联系本站删除!
DDR爱好者之家 Design By 杰米

《魔兽世界》大逃杀!60人新游玩模式《强袭风暴》3月21日上线

暴雪近日发布了《魔兽世界》10.2.6 更新内容,新游玩模式《强袭风暴》即将于3月21 日在亚服上线,届时玩家将前往阿拉希高地展开一场 60 人大逃杀对战。

艾泽拉斯的冒险者已经征服了艾泽拉斯的大地及遥远的彼岸。他们在对抗世界上最致命的敌人时展现出过人的手腕,并且成功阻止终结宇宙等级的威胁。当他们在为即将于《魔兽世界》资料片《地心之战》中来袭的萨拉塔斯势力做战斗准备时,他们还需要在熟悉的阿拉希高地面对一个全新的敌人──那就是彼此。在《巨龙崛起》10.2.6 更新的《强袭风暴》中,玩家将会进入一个全新的海盗主题大逃杀式限时活动,其中包含极高的风险和史诗级的奖励。

《强袭风暴》不是普通的战场,作为一个独立于主游戏之外的活动,玩家可以用大逃杀的风格来体验《魔兽世界》,不分职业、不分装备(除了你在赛局中捡到的),光是技巧和战略的强弱之分就能决定出谁才是能坚持到最后的赢家。本次活动将会开放单人和双人模式,玩家在加入海盗主题的预赛大厅区域前,可以从强袭风暴角色画面新增好友。游玩游戏将可以累计名望轨迹,《巨龙崛起》和《魔兽世界:巫妖王之怒 经典版》的玩家都可以获得奖励。