一区二区三区在线-一区二区三区亚洲视频-一区二区三区亚洲-一区二区三区午夜-一区二区三区四区在线视频-一区二区三区四区在线免费观看

服務器之家:專注于服務器技術及軟件下載分享
分類導航

PHP教程|ASP.NET教程|Java教程|ASP教程|編程技術|正則表達式|C/C++|IOS|C#|Swift|Android|VB|R語言|JavaScript|易語言|vb.net|

服務器之家 - 編程語言 - Java教程 - 詳解使用spring aop實現業務層mysql 讀寫分離

詳解使用spring aop實現業務層mysql 讀寫分離

2020-07-23 11:59馳馳的老爸 Java教程

本篇文章主要介紹了使用spring aop實現業務層mysql 讀寫分離,小編覺得挺不錯的,現在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧

spring aop , mysql 主從配置 實現讀寫分離,接下來把自己的配置過程,以及遇到的問題記錄下來,方便下次操作,也希望給一些朋友帶來幫助。

1.使用spring aop 攔截機制現數據源的動態選取。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
import java.lang.annotation.ElementType;
import java.lang.annotation.Target;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
/**
 * RUNTIME
 * 編譯器將把注釋記錄在類文件中,在運行時 VM 將保留注釋,因此可以反射性地讀取。
 * @author yangGuang
 *
 */
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface DataSource {
  String value();
}

3.利用Spring的AbstractRoutingDataSource解決多數據源的問題

?
1
2
3
4
5
6
7
8
9
10
import org.springframework.jdbc.datasource.lookup.AbstractRoutingDataSource;
 
 public class ChooseDataSource extends AbstractRoutingDataSource {
 
   @Override
   protected Object determineCurrentLookupKey() {
     return HandleDataSource.getDataSource();
   }
    
 }

4.利用ThreadLocal解決線程安全問題

?
1
2
3
4
5
6
7
8
9
10
public class HandleDataSource {
  public static final ThreadLocal<String> holder = new ThreadLocal<String>();
  public static void putDataSource(String datasource) {
    holder.set(datasource);
  }
   
  public static String getDataSource() {
    return holder.get();
  }  
}

5.定義一個數據源切面類,通過aop訪問,在spring配置文件中配置了,所以沒有使用aop注解。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
import java.lang.reflect.Method;
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;
import org.aspectj.lang.reflect.MethodSignature;
import org.springframework.stereotype.Component;
//@Aspect
//@Component
public class DataSourceAspect {
  //@Pointcut("execution(* com.apc.cms.service.*.*(..))") 
  public void pointCut(){}; 
   
 // @Before(value = "pointCut()")
   public void before(JoinPoint point)
    {
      Object target = point.getTarget();
      System.out.println(target.toString());
      String method = point.getSignature().getName();
      System.out.println(method);
      Class<?>[] classz = target.getClass().getInterfaces();
      Class<?>[] parameterTypes = ((MethodSignature) point.getSignature())
          .getMethod().getParameterTypes();
      try {
        Method m = classz[0].getMethod(method, parameterTypes);
        System.out.println(m.getName());
        if (m != null && m.isAnnotationPresent(DataSource.class)) {
          DataSource data = m.getAnnotation(DataSource.class);
          HandleDataSource.putDataSource(data.value());
        }
         
      } catch (Exception e) {
        e.printStackTrace();
      }
    }
}

6.配置applicationContext.xml

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
<!-- 主庫數據源 -->
 <bean id="writeDataSource" class="com.jolbox.bonecp.BoneCPDataSource" destroy-method="close">
  <property name="driverClass" value="com.mysql.jdbc.Driver"/>
  <property name="jdbcUrl" value="jdbc:mysql://172.22.14.6:3306/cpp?autoReconnect=true"/>
  <property name="username" value="root"/>
  <property name="password" value="root"/>
  <property name="partitionCount" value="4"/>
  <property name="releaseHelperThreads" value="3"/>
  <property name="acquireIncrement" value="2"/>
  <property name="maxConnectionsPerPartition" value="40"/>
  <property name="minConnectionsPerPartition" value="20"/>
  <property name="idleMaxAgeInSeconds" value="60"/>
  <property name="idleConnectionTestPeriodInSeconds" value="60"/>
  <property name="poolAvailabilityThreshold" value="5"/>
</bean>
 
<!-- 從庫數據源 -->
<bean id="readDataSource" class="com.jolbox.bonecp.BoneCPDataSource" destroy-method="close">
  <property name="driverClass" value="com.mysql.jdbc.Driver"/>
  <property name="jdbcUrl" value="jdbc:mysql://172.22.14.7:3306/cpp?autoReconnect=true"/>
  <property name="username" value="root"/>
  <property name="password" value="root"/>
  <property name="partitionCount" value="4"/>
  <property name="releaseHelperThreads" value="3"/>
  <property name="acquireIncrement" value="2"/>
  <property name="maxConnectionsPerPartition" value="40"/>
  <property name="minConnectionsPerPartition" value="20"/>
  <property name="idleMaxAgeInSeconds" value="60"/>
  <property name="idleConnectionTestPeriodInSeconds" value="60"/>
  <property name="poolAvailabilityThreshold" value="5"/>
</bean>
 
<!-- transaction manager, 事務管理 -->
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
  <property name="dataSource" ref="dataSource" />
</bean>
 
 
<!-- 注解自動載入 -->
<context:annotation-config />
 
<!--enale component scanning (beware that this does not enable mapper scanning!)-->
<context:component-scan base-package="com.apc.cms.persistence.rdbms" />
<context:component-scan base-package="com.apc.cms.service">
 <context:include-filter type="annotation"
    expression="org.springframework.stereotype.Component" /> 
</context:component-scan
 
<context:component-scan base-package="com.apc.cms.auth" />
 
<!-- enable transaction demarcation with annotations -->
<tx:annotation-driven />
 
 
<!-- define the SqlSessionFactory -->
<bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean">
  <property name="dataSource" ref="dataSource" />
  <property name="typeAliasesPackage" value="com.apc.cms.model.domain" />
</bean>
 
<!-- scan for mappers and let them be autowired -->
<bean class="org.mybatis.spring.mapper.MapperScannerConfigurer">
  <property name="basePackage" value="com.apc.cms.persistence" />
  <property name="sqlSessionFactory" ref="sqlSessionFactory" />
</bean>
 
<bean id="dataSource" class="com.apc.cms.utils.ChooseDataSource">
  <property name="targetDataSources"
     <map key-type="java.lang.String"
       <!-- write -->
       <entry key="write" value-ref="writeDataSource"/> 
       <!-- read -->
       <entry key="read" value-ref="readDataSource"/> 
     </map
      
  </property
  <property name="defaultTargetDataSource" ref="writeDataSource"/> 
</bean>
  
<!-- 激活自動代理功能 -->
<aop:aspectj-autoproxy proxy-target-class="true"/>
 
<!-- 配置數據庫注解aop -->
<bean id="dataSourceAspect" class="com.apc.cms.utils.DataSourceAspect" />
<aop:config>
  <aop:aspect id="c" ref="dataSourceAspect">
    <aop:pointcut id="tx" expression="execution(* com.apc.cms.service..*.*(..))"/>
    <aop:before pointcut-ref="tx" method="before"/>
  </aop:aspect>
</aop:config>
<!-- 配置數據庫注解aop -->

7.使用注解,動態選擇數據源,分別走讀庫和寫庫。

?
1
2
3
4
5
6
7
8
9
@DataSource("write")
public void update(User user) {
  userMapper.update(user);
}
 
@DataSource("read")
public Document getDocById(long id) {
  return documentMapper.getById(id);
}

測試寫操作:可以通過應用修改數據,修改主庫數據,發現從庫的數據被同步更新了,所以定義的write操作都是走的寫庫

測試讀操作:  后臺修改從庫數據,查看主庫的數據沒有被修改,在應用頁面中刷新,發現讀的是從庫的數據,說明讀寫分離ok。

遇到的問題總結:

問題1:項目是maven工程,用到了Spring aop機制,除了spring的核心jar包以為,還需要用到的jar包有aspectj.jar,aspectjweaver.jar,aopalliance.jar查看項目中的pom,發現缺少依賴包,由于本地倉庫沒有這些jar,查找可以提供下載jar包的maven中央庫庫,配置到maven中,自動更新:

?
1
2
3
4
5
6
<repository>
   <id>nexus</id>
   <name>nexus</name>
   <url>http://repository.sonatype.org/content/groups/public/</url>
   <layout>default</layout>
 </repository>

配置項目依賴的jar,主要是缺少這兩個。

?
1
2
3
4
5
6
7
8
9
10
  <dependency>
    <groupId>aspectj</groupId>
    <artifactId>aspectjrt</artifactId>
    <version>1.5.4</version>
</dependency>
<dependency>
    <groupId>aspectj</groupId>
    <artifactId>aspectjweaver</artifactId>
    <version>1.5.4</version>
lt;/dependency>

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持服務器之家。

原文鏈接:http://blog.csdn.net/huoyunshen88/article/details/36674861

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 欧美一级专区免费大片俄罗斯 | 俄罗斯妈妈k8影院在线观看 | 女人叉开腿让男人捅 | 毛片在线播放a | 免费人成黄页在线观看69 | 高清日韩在线 | 国产精品66福利在线观看 | 国产综合视频在线 | 精品suv一区二区三区 | 日本艳鉧动漫1~6在线观看 | 黑人巨| 国产播放器一区 | www日本在线观看 | 亚洲国产成人久久精品hezyo | 岛国a香蕉片不卡在线观看 荡女淫春2古装 | 亚洲美洲国产日产 | 果冻传媒和91制片厂网站软件 | 欧美精品v日韩精品v国产精品 | 日本性生活大片 | 日韩人成 | 国产精品青青青高清在线密亚 | 国内自拍网红在线综合 | 国产精品久久久久久久人人看 | 日韩中文字幕一区 | 色先锋 影音先锋a 资源站 | 高黄h文各种play | 亚洲AV久久无码精品九九软件 | 亚洲精品动漫免费二区 | 鬼惨笑小说 | 九九国产在线观看 | 午夜大片在线观看 | 精品9e精品视频在线观看 | 国产成人精品日本亚洲网站 | 午夜影院一区二区三区 | 国产精品亚洲片夜色在线 | 国产精品露脸国语对白手机视频 | 国产自拍资源 | 精选国产AV精选一区二区三区 | 亚洲欧美日韩成人一区在线 | 国产自拍影院 | 亚洲激情综合 |