溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

SpringBoot中怎么實(shí)現(xiàn)一個(gè)讀寫分離組件

發(fā)布時(shí)間:2021-07-08 16:57:37 來源:億速云 閱讀:168 作者:Leah 欄目:web開發(fā)

這期內(nèi)容當(dāng)中小編將會(huì)給大家?guī)碛嘘P(guān)SpringBoot中怎么實(shí)現(xiàn)一個(gè)讀寫分離組件,文章內(nèi)容豐富且以專業(yè)的角度為大家分析和敘述,閱讀完這篇文章希望大家可以有所收獲。

1.pom.xml配置文件

<dependency>             <groupId>org.springframework.boot</groupId>             <artifactId>spring-boot-starter</artifactId> </dependency> <dependency>             <groupId>org.springframework.boot</groupId>             <artifactId>spring-boot-starter-data-jpa</artifactId> </dependency> <dependency>             <groupId>org.springframework.boot</groupId>             <artifactId>spring-boot-configuration-processor</artifactId>             <optional>true</optional> </dependency>

 2.application.yml配置文件

pack:   datasource:     pointcut: execution(public * net.greatsoft.service.base.*.*(..)) || execution(public * net.greatsoft.service.xxx.*.*(..))     master:       driverClassName: oracle.jdbc.driver.OracleDriver       jdbcUrl: jdbc:oracle:thin:@10.100.102.113:1521/orcl       username: test       password: test       minimumIdle: 10       maximumPoolSize: 200       autoCommit: true       idleTimeout: 30000       poolName: MbookHikariCP       maxLifetime: 1800000       connectionTimeout: 30000       connectionTestQuery: SELECT 1 FROM DUAL       slaves:       - driverClassName: oracle.jdbc.driver.OracleDriver         jdbcUrl: jdbc:oracle:thin:@10.100.102.113:1521/orcl         username: dc         password: dc         minimumIdle: 10         maximumPoolSize: 200         autoCommit: true         idleTimeout: 30000         poolName: MbookHikariCP         maxLifetime: 1800000         connectionTimeout: 30000         connectionTestQuery: SELECT 1 FROM DUAL       - driverClassName: oracle.jdbc.driver.OracleDriver         jdbcUrl: jdbc:oracle:thin:@10.100.102.113:1521/orcl         username: empi         password: empi         minimumIdle: 10         maximumPoolSize: 200         autoCommit: true         idleTimeout: 30000         poolName: MbookHikariCP         maxLifetime: 1800000         connectionTimeout: 30000         connectionTestQuery: SELECT 1 FROM DUAL

pointcut:定義切點(diǎn),那些方法是需要攔截(從讀庫中操作)。

master:寫庫配置。

slaves:讀庫配置(List集合)。

3.屬性配置類

@Component @ConfigurationProperties(prefix = "pack.datasource") public class RWDataSourceProperties {          private String pointcut ;     private HikariConfig master ;     private List<HikariConfig> slaves = new ArrayList<>();      }

 4.讀寫配置類

public class RWConfig  {          private static Logger logger = LoggerFactory.getLogger(RWConfig.class) ;      @Bean     public HikariDataSource masterDataSource(RWDataSourceProperties rwDataSourceProperties) {         return new HikariDataSource(rwDataSourceProperties.getMaster()) ;     }          @Bean     public List<HikariDataSource> slaveDataSources(RWDataSourceProperties rwDataSourceProperties) {         List<HikariDataSource> lists = new ArrayList<>() ;         for(HikariConfig config : rwDataSourceProperties.getSlaves()) {             lists.add(new HikariDataSource(config)) ;         }         return lists ;     }          @Bean   @Primary     @DependsOn({"masterDataSource", "slaveDataSources"})     public AbstractRoutingDataSource routingDataSource(@Qualifier("masterDataSource")DataSource masterDataSource,             @Qualifier("slaveDataSources")List<HikariDataSource> slaveDataSources) {         BaseRoutingDataSource ds = new BaseRoutingDataSource() ;         Map<Object, Object> targetDataSources = new HashMap<>(2) ;         targetDataSources.put("master", masterDataSource) ;         for (int i = 0; i < slaveDataSources.size(); i++) {             targetDataSources.put("slave-" + i, slaveDataSources.get(i)) ;         }         ds.setDefaultTargetDataSource(masterDataSource) ;         ds.setTargetDataSources(targetDataSources) ;         return ds ;     }      }

 5.數(shù)據(jù)源路由

  1. public class BaseRoutingDataSource extends AbstractRoutingDataSource { 

  2.  

  3.     @Resource 

  4.     private DataSourceHolder holder; 

  5.      

  6.     @Override 

  7.     protected Object determineCurrentLookupKey() { 

  8.         return holder.get() ; 

  9.     } 

  10.      


public class DataSourceHolder {          private ThreadLocal<Integer> context = new ThreadLocal<Integer>() {         @Override         protected Integer initialValue() {             return 0 ;         }     };          @Resource     private BaseSlaveLoad slaveLoad ;          public String get() {         Integer type = context.get() ;         return type == null || type == 0 ? "master" : "slave-" + slaveLoad.load() ;     }          public void set(Integer type) {         context.set(type) ;     }      }

通過aop動(dòng)態(tài)設(shè)置context的內(nèi)容值,0為從寫庫中操作,其它的都在讀庫中操作。

BaseSlaveLoad類為到底從那個(gè)讀庫中選取的一個(gè)算法類,默認(rèn)實(shí)現(xiàn)使用的是輪詢算法。

  1. public interface BaseSlaveLoad { 

  2.  

  3.     int load() ; 

  4.      


public abstract class AbstractSlaveLoad implements BaseSlaveLoad {      @Resource     protected List<HikariDataSource> slaveDataSources ;      }

這里定義一個(gè)抽象類注入了讀庫列表,所有的實(shí)現(xiàn)類從該類中繼承即可。

public class PollingLoad extends AbstractSlaveLoad {          private int index = 0 ;     private int size = 1 ;          @PostConstruct     public void init() {         size = slaveDataSources.size() ;     }          @Override     public int load() {         int n = index ;         synchronized (this) {             index = (++index) % size ;         }         return n ;     }      }

配置成Bean

@Bean     @ConditionalOnMissingBean     public BaseSlaveLoad slaveLoad() {         return new PollingLoad() ;     }          @Bean     public DataSourceHolder dataSourceHolder() {         return new DataSourceHolder() ;     }

 6.數(shù)據(jù)源AOP

public class DataSourceAspect implements MethodInterceptor {      private DataSourceHolder holder ;          public DataSourceAspect(DataSourceHolder holder) {         this.holder = holder ;     }          @Override     public Object invoke(MethodInvocation invocation) throws Throwable {         Method method = invocation.getMethod() ;         String methodName = method.getName() ;         SlaveDB slaveDB = method.getAnnotation(SlaveDB.class) ;         if (slaveDB == null) {             slaveDB = method.getDeclaringClass().getAnnotation(SlaveDB.class) ;         }         if (methodName.startsWith("find")                  || methodName.startsWith("get")                 || methodName.startsWith("query")                 || methodName.startsWith("select")                 || methodName.startsWith("list")                 || slaveDB != null) {             holder.set(1) ;         } else {             holder.set(0) ;         }         return invocation.proceed();     }  }

應(yīng)該切點(diǎn)需要?jiǎng)討B(tài)配置,所以這里采用spring aop的方式來配置

@Bean     public AspectJExpressionPointcutAdvisor logAdvisor(RWDataSourceProperties props, DataSourceHolder holder) {         AspectJExpressionPointcutAdvisor advisor = new AspectJExpressionPointcutAdvisor() ;         logger.info("執(zhí)行表達(dá)式:{}", props.getPointcut()) ;         advisor.setExpression(props.getPointcut()) ;         advisor.setAdvice(new DataSourceAspect(holder)) ;         return advisor ;     }

 7.Enable開啟功能

public class RWImportSelector implements ImportSelector {      @Override     public String[] selectImports(AnnotationMetadata importingClassMetadata) {         return new String[] {RWConfig.class.getName()} ;     }  }

這里的RWConfig為我們上面的配置類

  1. @Retention(RetentionPolicy.RUNTIME) 

  2. @Target(ElementType.TYPE) 

  3. @Documented 

  4. @Import({RWImportSelector.class}) 

  5. public @interface EnableRW { 


@Documented @Retention(RUNTIME) @Target({ TYPE, METHOD }) public @interface SlaveDB { }

有@SlaveDB的注解方法會(huì)類都會(huì)從讀庫中操作。

到此讀寫分離組件開發(fā)完成。

8.打包安裝到本地倉庫

mvn install -Dmaven.test.skip=true

9.新建base-web項(xiàng)目

引入依賴

<dependency>             <groupId>com.pack</groupId>             <artifactId>xg-component-rw</artifactId>             <version>1.0.0</version> </dependency>

啟動(dòng)類添加注解開啟讀寫分離功能

@SpringBootApplication @EnableRW public class BaseWebApplication {      public static void main(String[] args) {         SpringApplication.run(BaseWebApplication.class, args);     }  }

測(cè)試:

第一次查詢:

SpringBoot中怎么實(shí)現(xiàn)一個(gè)讀寫分離組件

第二次查詢:

SpringBoot中怎么實(shí)現(xiàn)一個(gè)讀寫分離組件

為了區(qū)別兩個(gè)從庫設(shè)置不同的數(shù)據(jù)

SpringBoot中怎么實(shí)現(xiàn)一個(gè)讀寫分離組件

上述就是小編為大家分享的SpringBoot中怎么實(shí)現(xiàn)一個(gè)讀寫分離組件了,如果剛好有類似的疑惑,不妨參照上述分析進(jìn)行理解。如果想知道更多相關(guān)知識(shí),歡迎關(guān)注億速云行業(yè)資訊頻道。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI