背景
マスター/スレーブデータベースを構成した後、コードレベルで読み取りと書き込みの分離を実現するにはどうすればよいでしょうか?
ユーザー定義のデータベースルーティング
Spring Boot は、ユーザー定義のルールに従って現在のデータベースを選択する AbstractRoutingDataSource を提供します。これにより、クエリを実行する前に読み取りスレーブ データベースを設定し、実行後にメイン データベースに復元できます。完成しました。
動的にルーティング可能なデータソースを実装し、各データベースクエリ操作の前に実行します
ReadWriteSplitRoutingDataSource.java
import org.springframework.jdbc.datasource.lookup.AbstractRoutingDataSource; /** * @author songrgg * @since 1.0 */ public class ReadWriteSplitRoutingDataSource extends AbstractRoutingDataSource { @Override protected Object determineCurrentLookupKey() { return DbContextHolder.getDbType(); } }
ReadWriteSplitRoutingDataSourceの動的読み取り構成に使用されるスレッドプライベートルーティング構成
DbContextHolder.java
/** * @author songrgg * @since 1.0 */ public class DbContextHolder { public enum DbType { MASTER, SLAVE } private static final ThreadLocal<DbType> contextHolder = new ThreadLocal<>(); public static void setDbType(DbType dbType) { if(dbType == null){ throw new NullPointerException(); } contextHolder.set(dbType); } public static DbType getDbType() { return contextHolder.get() == null ? DbType.MASTER : contextHolder.get(); } public static void clearDbType() { contextHolder.remove(); } }
AOPコード
が使用する最適化AOP を使用して、データベースのセットアップ操作をコードから分離します。ここでの粒度はメソッド レベルで制御されるため、このメソッドに関与するデータベース トランザクションを読み取り専用としてマークし、ライブラリから取得するためにアノテーションが使用されます。
読み取り専用アノテーション。メソッドにアノテーションを付けるために使用されるデータベース操作は、スレーブ データベースにのみ送信されます。
ReadOnlyConnection.java
package com.wallstreetcn.hatano.config; import java.lang.annotation.ElementType; import java.lang.annotation.Retention; import java.lang.annotation.RetentionPolicy; import java.lang.annotation.Target; /** * Indicates the database operations is bound to the slave database. * AOP interceptor will set the database to the slave with this interface. * @author songrgg * @since 1.0 */ @Target({ElementType.METHOD, ElementType.TYPE}) @Retention(RetentionPolicy.RUNTIME) public @interface ReadOnlyConnection { }
ReadOnlyConnectionInterceptor.java
import org.aspectj.lang.ProceedingJoinPoint; import org.aspectj.lang.annotation.Around; import org.aspectj.lang.annotation.Aspect; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.core.Ordered; import org.springframework.stereotype.Component; /** * Intercept the database operations, bind database to read-only database as this annotation * is applied. * @author songrgg * @since 1.0 */ @Aspect @Component public class ReadOnlyConnectionInterceptor implements Ordered { private static final Logger logger = LoggerFactory.getLogger(ReadOnlyConnectionInterceptor.class); @Around("@annotation(readOnlyConnection)") public Object proceed(ProceedingJoinPoint proceedingJoinPoint, ReadOnlyConnection readOnlyConnection) throws Throwable { try { logger.info("set database connection to read only"); DbContextHolder.setDbType(DbContextHolder.DbType.SLAVE); Object result = proceedingJoinPoint.proceed(); return result; } finally { DbContextHolder.clearDbType(); logger.info("restore database connection"); } } @Override public int getOrder() { return 0; } }
UserService.java
@ReadOnlyConnection public List<User> getUsers(Integer page, Integer limit) { return repository.findAll(new PageRequest(page, limit)); }
Druidデータベース接続プールを構成する
build.gradle
compile("com.alibaba:druid:1.0.18")
Groovy 依存関係の挿入
dataSource をルーティング可能なデータ ソースとして構成する
context.groovy
import com.alibaba.druid.pool.DruidDataSource import DbContextHolder import ReadWriteSplitRoutingDataSource ** SOME INITIALIZED CODE LOAD PROPERTIES ** def dataSourceMaster = new DruidDataSource() dataSourceMaster.url = properties.get('datasource.master.url') println("master set to " + dataSourceMaster.url) dataSourceMaster.username = properties.get('datasource.master.username') dataSourceMaster.password = properties.get('datasource.master.password') def dataSourceSlave = new DruidDataSource() dataSourceSlave.url = properties.get('datasource.slave.url') println("slave set to " + dataSourceSlave.url) dataSourceSlave.username = properties.get('datasource.slave.username') dataSourceSlave.password = properties.get('datasource.slave.password') beans { dataSource(ReadWriteSplitRoutingDataSource) { bean -> targetDataSources = [ (DbContextHolder.DbType.MASTER): dataSourceMaster, (DbContextHolder.DbType.SLAVE): dataSourceSlave ] } }
以上がこの記事の全内容です。皆様の学習に役立つことを願っております。また、皆様も PHP 中国語をサポートしていただければ幸いです。 Webサイト。
データベースの読み取りと書き込みの分離を実装する Spring Boot の方法に関連するその他の記事については、PHP 中国語 Web サイトに注目してください。