国产片侵犯亲女视频播放_亚洲精品二区_在线免费国产视频_欧美精品一区二区三区在线_少妇久久久_在线观看av不卡

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

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

服務器之家 - 編程語言 - Java教程 - 詳解Spring AOP 實現主從讀寫分離

詳解Spring AOP 實現主從讀寫分離

2020-08-23 14:30Mafly Java教程

本篇文章主要介紹了Spring AOP 實現主從讀寫分離,小編覺得挺不錯的,現在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧

深刻討論為什么要讀寫分離?

為了服務器承載更多的用戶?提升了網站的響應速度?分攤數據庫服務器的壓力?就是為了雙機熱備又不想浪費備份服務器?上面這些回答,我認為都不是錯誤的,但也都不是完全正確的。「讀寫分離」并不是多么神奇的東西,也帶不來多么大的性能提升,也許更多的作用的就是數據安全的備份吧。

從一個庫到讀寫分離,從理論上對服務器壓力來說是會帶來一倍的性能提升,但你仔細思考一下,你的應用服務器真的很需要這一倍的提升么?那倒不如你去試著在服務器使用一下緩存系統,如 Memcached、Redis 這些分布式緩存,那性能可能是幾十倍的提升。而且,在服務器硬件異常強悍及性能廉價的今天,完全更沒必要了,所以,在今天,我認為它更多的職責就是為了數據安全而設計的,同時又提升了一些性能,這樣也挺好。

可能我們更應該稱之為主從分離

利用 AOP 實現讀寫分離

讀寫分離方式很簡單,就是在你讀數據是去連接從庫,在你寫數據的時候去連接主庫,具體代碼實現當然就是連接時候去操作了,這沒什么難度,在代碼里寫就是了。可是,有追求的程序猿都是不是這么解決問題的呢!

 其實通過上篇的 Spring AOP 攔截器的基本實現 我們知道 AOP 可以實現在方法開始執行前后插入執行我們想要的代碼,那這樣,我們是不是可以在執行數據庫操作前根據業務來動態切換數據源呢?

思考一下這個方式理論上好像是可行的,這種方式首先不需要在業務代碼中去做切換,二是可能以后我們不需要讀寫分離了,把 AOP 切換的代碼去掉就行了,三是可能就是拓展性好了。

等不了了,開始擼代碼

你可能想深入的了解的話,我這里給你幾個程序里用到的關鍵字enum(枚舉)annotation(自定義注解)JoinPoint(注入點)AbstractRoutingDataSource(數據源接口子類),你理解了這些就知道了,其實你并不需要深入某些深層的東西,了解下即可。

一、建立JdbcContextHolder.java類

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
public class JdbcContextHolder {
 
 private static final ThreadLocal<String> contextHolder = new ThreadLocal<String>();
 
 public static void setJdbcType(String jdbcType) {
 contextHolder.set(jdbcType);
 }
 
 public static void setSlave() {
 setJdbcType("slave");
 }
 
 public static void setMaster() {
 clearJdbcType();
 }
 
 public static String getJdbcType() {
 return (String) contextHolder.get();
 }
 
 public static void clearJdbcType() {
 contextHolder.remove();
 }
}

這個類的作用就是用來設置、獲取數據源連接

二、新建DynamicDataSource.java類,繼承于AbstractRoutingDataSource

?
1
2
3
4
5
6
7
8
9
10
11
12
import org.springframework.jdbc.datasource.lookup.AbstractRoutingDataSource;
 
import cn.mayongfa.common.JdbcContextHolder;
 
public class DynamicDataSource extends AbstractRoutingDataSource {
 
 @Override
 protected Object determineCurrentLookupKey() {
 // 獲取當前數據源連接
 return JdbcContextHolder.getJdbcType();
 }
}

通過研究,我們知道determineCurrentLookupKey方法是獲取相關數據源連接的,所以重寫determineCurrentLookupKey方法就可以啦,然后我們去通過剛剛我們建立的JdbcContextHolder類去獲取。那怎么設置呢?

三、建立數據源DataSourceType.java枚舉類

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
public enum DataSourceType {
 
 //主庫
 Master("master"),
 
 //從庫
 Slave("slave");
 
 private DataSourceType(String name) {
 this.name = name;
 }
 
 private String name;
 
 public String getName() {
 return name;
 }
 public void setName(String name) {
 this.name = name;
 }
}

這個枚舉類的作用其實就是為了設置數據源而生的,它的目的就是讓設置數據源時更方便,如絲般順滑。

四、新建DataSource.java Annotation(自定義注解)類

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
import java.lang.annotation.Documented;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
 
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
@Documented
public @interface DataSource {
 
 DataSourceType value() default DataSourceType.Master;
 
}

自定義注解的意義不再過多討論,一句話來說就是可以讓你在類或方法名上以打標簽的形式讓該方法變得不一樣。具體怎么「不一樣」,這個在于你。

五、新建DataSourceChoose.java數據庫切換類

?
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
import java.lang.reflect.Method;
 
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.reflect.MethodSignature;
 
import cn.mayongfa.common.JdbcContextHolder;
 
public class DataSourceChoose {
 
//方法執行前
public void before(JoinPoint point){
 Object target = point.getTarget();
 String method = point.getSignature().getName();
 Class<?>[] classz = target.getClass().getInterfaces();
 MethodSignature methodSignature = (MethodSignature)point.getSignature();
 Class<?>[] parameterTypes = methodSignature.getMethod().getParameterTypes();
 try {
  Method m = classz[0].getMethod(method, parameterTypes);
  if (m!=null && m.isAnnotationPresent(DataSource.class)) {
  DataSource data = m.getAnnotation(DataSource.class);
  JdbcContextHolder.clearJdbcType();
  JdbcContextHolder.setJdbcType(data.value().getName());
  }
 } catch (Exception e) {
  // TODO: handle exception
 }
}
}

這個其實是一個攔截器類,主要作用就是攔截那些方法名上有@DataSource這個自定義注解的,完了根據獲取注解的value()值,來做相應的數據源切換。

到這里,整個讀寫分離的分析及業務邏輯和具體代碼都完了,以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持服務器之家。

原文鏈接:http://www.cnblogs.com/mafly/p/master_slave.html

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 国产精品网站在线观看 | 国产精品免费视频一区 | 懂色av成人一区二区三区 | 欧美在线亚洲 | 亚洲综合激情 | 可以免费在线观看av的网站 | 欧美性久久 | 黄色的视频免费看 | 国产日韩精品一区二区 | 欧美久久久网站 | 精品国产一级 | 国产激情在线观看 | 亚洲 欧美 另类 综合 偷拍 | 一区在线观看 | 欧美日韩一区二区三区在线观看 | av网站免费看 | 亚洲电影在线观看 | 蜜桃传媒一区二区 | 亚洲国产视频一区 | 欧美日韩国产一区二区三区不卡 | 男人的天堂中文字幕 | 精品国产乱码久久久久久久软件 | 操操操干干 | 黄色小视频在线观看 | 中文字幕一区二区三区在线视频 | 天天操网 | 日本全黄裸体片 | 亚洲国产视频一区 | 午夜看片 | 亚洲精品二区 | 日韩黄色片免费看 | 亚洲精品久久久久久国 | 国产亚洲一区二区三区 | 91无吗 | 午夜精品久久 | 成人精品一区二区三区中文字幕 | 手机黄网www8xcn | 亚洲福利二区 | 精品国产乱码久久久久久牛牛 | 国产精品久久久久久久9999 | 中文字幕免费 |