【SpringBoot高级篇】SpringBoot集成Mybatis实现多数据源配置+跨数据源事务
admin
2024-02-16 21:09:10
0

【SpringBoot高级篇】SpringBoot集成Mybatis实现多数据源配置+跨数据源事务

  • Pom依赖
  • application.yml
  • 多数据源配置
    • MasterDataSourceConfig
    • ClusterDataSourceConfig
    • 启动类
    • 使用
  • 实现跨数据源事务

开发中经常有这样的需要: 读写分离。微服务环境下可以实现一个服务读取一个数据库,另一个服务写库。但是在实际应用中有时也需要在一个服务中读写不同的数据库。可以在一个SpringBoot单体项目中配置多个数据源解决读写库分离。

Pom依赖

org.springframework.bootspring-boot-starter-parent2.2.2.RELEASE

org.springframework.bootspring-boot-starter-jdbcmysqlmysql-connector-javacom.alibabadruid-spring-boot-starter1.2.14org.springframework.bootspring-boot-starter-testorg.mybatis.spring.bootmybatis-spring-boot-starter2.2.2com.github.pagehelperpagehelper-spring-boot-starter1.3.0org.springframework.bootspring-boot-starter-aop

application.yml

# 数据源
spring:datasource:# master主数据源配置master:driverClassName: com.mysql.cj.jdbc.Driverurl: jdbc:mysql://localhost:3306/ds1?characterEncoding=UTF-8&serverTimezone=GMT%2B8username: rootpassword: root# cluster从数据源配置cluster:driverClassName: com.mysql.cj.jdbc.Driverurl: jdbc:mysql://localhost:3306/ds2?characterEncoding=UTF-8&serverTimezone=GMT%2B8username: rootpassword: root#pagehelper分页插件配置
pagehelper:master:helper-dialect: mysql  #设置数据库类型reasonable: true  #开启合理化:页码<=0 查询第一页,页码>=总页数查询最后一页support-methods-arguments: true  #支持通过 Mapper 接口参数来传递分页参数params: count=countsql

多数据源配置

Java代码方式创建主从数据源配置。 使用@Primary: 用于指定bean的注入优先级。被@Primary修饰的bean对象优先注入

MasterDataSourceConfig

/*** @ClassName: MasterDataSourceConfig* @Description: mysql主库配置类*/
@Configuration
@MapperScan(basePackages = MasterDataSourceConfig.PACKAGE,sqlSessionTemplateRef = "masterSqlSessionTemplate")
public class MasterDataSourceConfig {/*** 设置扫描包的路径*/public static final String PACKAGE = "cn.zysheep.dao.master";/*** Mapper配置文件路径*/public static final String MAPPER_LOCATION = "classpath*:mapper/master/*.xml";/*** 实体别名*/public static final String ENTITY_ALIASES = "cn.zysheep.entity";@Value("${spring.datasource.master.driverClassName}")private String driverClassName;@Value("${spring.datasource.master.url}")private String url;@Value("${spring.datasource.master.username}")private String user;@Value("${spring.datasource.master.password}")private String password;@Value("${pagehelper.master.helper-dialect}")private String helperDialect;@Value("${pagehelper.master.reasonable}")private String reasonable;/*** 创建数据源* @return DataSource* @Primary: 用于指定bean的注入优先级。被@Primary修饰的bean对象优先注入*/@Bean(name = "masterDataSource")@Primarypublic DataSource masterDataSource() {DruidDataSource dataSource = new DruidDataSource();dataSource.setDriverClassName(driverClassName);dataSource.setUrl(url);dataSource.setUsername(user);dataSource.setPassword(password);return dataSource;}/*** 创建工厂*@param dataSource*@throws Exception*@return SqlSessionFactory*/@Bean(name = "masterSqlSessionFactory")@Primarypublic SqlSessionFactory masterSqlSessionFactory(@Qualifier("masterDataSource") DataSource dataSource) throws Exception {SqlSessionFactoryBean sessionFactory = new SqlSessionFactoryBean();sessionFactory.setDataSource(dataSource);sessionFactory.setMapperLocations(new PathMatchingResourcePatternResolver().getResources(MAPPER_LOCATION));sessionFactory.setTypeAliasesPackage(ENTITY_ALIASES);org.apache.ibatis.session.Configuration configuration = new org.apache.ibatis.session.Configuration();// 字段值为空也可以插入configuration.setJdbcTypeForNull(JdbcType.NULL);configuration.setCallSettersOnNulls(true);// mybatis logconfiguration.setLogImpl(StdOutImpl.class);sessionFactory.setConfiguration(configuration);PageInterceptor pageInterceptor = new PageInterceptor();// 分页配置Properties p = new Properties();p.setProperty("helperDialect", helperDialect);p.setProperty("reasonable", reasonable);pageInterceptor.setProperties(p);sessionFactory.setPlugins(pageInterceptor);return sessionFactory.getObject();}/*** 创建事务*@param dataSource*@return DataSourceTransactionManager*/@Bean(name = "masterTransactionManager")@Primarypublic DataSourceTransactionManager masterDataSourceTransactionManager(@Qualifier("masterDataSource") DataSource dataSource) {return new DataSourceTransactionManager(dataSource);}/*** 创建模板*@param sqlSessionFactory*@return SqlSessionTemplate*/@Bean(name = "masterSqlSessionTemplate")@Primarypublic SqlSessionTemplate masterSqlSessionTemplate(@Qualifier("masterSqlSessionFactory") SqlSessionFactory sqlSessionFactory) {return new SqlSessionTemplate(sqlSessionFactory);}
}

ClusterDataSourceConfig

/*** @ClassName: DataSourceConfig* @Description: mysql从库配置类*/
@Configuration
@MapperScan(basePackages = ClusterDataSourceConfig.PACKAGE,sqlSessionTemplateRef = "clusterSqlSessionTemplate")
public class ClusterDataSourceConfig {/*** 设置扫描包的路径*/public static final String PACKAGE = "cn.zysheep.dao.cluster";/*** Mapper配置文件路径*/public static final String MAPPER_LOCATION = "classpath*:mapper/cluster/*.xml";/*** 实体别名*/public static final String ENTITY_ALIASES = "cn.zysheep.entity";@Value("${spring.datasource.cluster.driverClassName}")private String driverClassName;@Value("${spring.datasource.cluster.url}")private String url;@Value("${spring.datasource.cluster.username}")private String user;@Value("${spring.datasource.cluster.password}")private String password;@Value("${pagehelper.master.helper-dialect}")private String helperDialect;@Value("${pagehelper.master.reasonable}")private String reasonable;/*** 创建数据源*@return DataSource*/@Bean(name = "clusterDataSource")public DataSource masterDataSource() {DruidDataSource dataSource = new DruidDataSource();dataSource.setDriverClassName(driverClassName);dataSource.setUrl(url);dataSource.setUsername(user);dataSource.setPassword(password);return dataSource;}/*** 创建工厂*@param dataSource*@throws Exception*@return SqlSessionFactory*/@Bean(name = "clusterSqlSessionFactory")public SqlSessionFactory masterSqlSessionFactory(@Qualifier("clusterDataSource") DataSource dataSource) throws Exception {SqlSessionFactoryBean sessionFactory = new SqlSessionFactoryBean();sessionFactory.setDataSource(dataSource);sessionFactory.setMapperLocations(new PathMatchingResourcePatternResolver().getResources(MAPPER_LOCATION));sessionFactory.setTypeAliasesPackage(ENTITY_ALIASES);org.apache.ibatis.session.Configuration configuration = new org.apache.ibatis.session.Configuration();// 字段值为空也可以插入configuration.setJdbcTypeForNull(JdbcType.NULL);configuration.setCallSettersOnNulls(true);// mybatis logconfiguration.setLogImpl(StdOutImpl.class);sessionFactory.setConfiguration(configuration);PageInterceptor pageInterceptor = new PageInterceptor();// 分页配置Properties p = new Properties();p.setProperty("helperDialect", helperDialect);p.setProperty("reasonable", reasonable);pageInterceptor.setProperties(p);sessionFactory.setPlugins(pageInterceptor);return sessionFactory.getObject();}/*** 创建事务管理器*@param dataSource*@return DataSourceTransactionManager*/@Bean(name = "clusterTransactionManager")public DataSourceTransactionManager masterDataSourceTransactionManager(@Qualifier("clusterDataSource") DataSource dataSource) {return new DataSourceTransactionManager(dataSource);}/*** 创建模板*@param sqlSessionFactory*@return SqlSessionTemplate*/@Bean(name = "clusterSqlSessionTemplate")public SqlSessionTemplate masterSqlSessionTemplate(@Qualifier("clusterSqlSessionFactory") SqlSessionFactory sqlSessionFactory) {return new SqlSessionTemplate(sqlSessionFactory);}
}

启动类

启动类添加@EnableTransactionManagement注解,开启注解事务

@EnableTransactionManagement
@SpringBootApplication(exclude = PageHelperAutoConfiguration.class)
public class MultiDataSourceApplication {public static void main(String[] args) {SpringApplication.run(MultiDataSourceApplication.class, args);}
}

使用

在service层类或者方法上添加@Transactional注解。并指定属性事务管理器transactionManager的值,根据使用不同的数据源,选择不同的数据事务管理器

@Transactional(transactionManager = "masterTransactionManager")
public void insertUser() {// TODO 添加用户业务操作
}@Transactional(transactionManager = "clusterTransactionManager")
public void listUser() {// TODO 查询用户业务操作
}

思考: @Transactional只能指定一个事务管理器,并且注解不允许重复,所以就只能使用一个数据源的事务管理器了。那么对于一个方法涉及到多个数据源操作需要保证事务一致性的怎么办呢?

实现跨数据源事务

基于注解+AOP使用编程式事务的方式实现跨数据源事务

@Target({ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
public @interface MultiDataSourceTransactional {/*** 事务管理器数组*/String[] transactionManagers();
}

首先多个数据源的事务分别都开起来,然后各事务分别去执行对应的sql(此所谓第一阶段提交),最后如果都成功就把事务全部提交,只要有一个失败就把事务都回滚——此所谓第二阶段提交。

/*** 需要注意的点:* 1)声明事务和提交事务或者回滚事务的顺序应该相反的,就是先进后出,所以采用了栈来存储。* 2)线程执行结束后记得清空本地变量。* 3)可以参照@Transactional的那些属性升级功能,比如隔离级别回滚异常等。*/
@Component
@Aspect
public class MultiTransactionAop {/*** 线程本地变量:为什么使用栈?※为了达到后进先出的效果※*/private static final ThreadLocal>> THREAD_LOCAL = new ThreadLocal<>();private static Logger logger = LoggerFactory.getLogger(MultiTransactionAop.class);/*** 用于获取事务管理器*/@Autowiredprivate ApplicationContext applicationContext;/*** 事务声明*/private DefaultTransactionDefinition def = new DefaultTransactionDefinition();{// 非只读模式def.setReadOnly(false);// 事务隔离级别:采用数据库的def.setIsolationLevel(TransactionDefinition.ISOLATION_DEFAULT);// 事务传播行为def.setPropagationBehavior(TransactionDefinition.PROPAGATION_REQUIRED);}/*** 切点*/@Pointcut("@annotation(cn.zysheep.datasource.anntation.MultiDataSourceTransactional)")public void pointcut() {}/*** 声明事务** @param transactional 注解*/@Before("pointcut() && @annotation(transactional)")public void before(MultiDataSourceTransactional transactional) {// 根据设置的事务名称按顺序声明,并放到ThreadLocal里String[] transactionManagerNames = transactional.transactionManagers();Stack> pairStack = new Stack<>();for (String transactionManagerName : transactionManagerNames) {DataSourceTransactionManager transactionManager = applicationContext.getBean(transactionManagerName, DataSourceTransactionManager.class);TransactionStatus transactionStatus = transactionManager.getTransaction(def);Map transactionMap = new HashMap<>();transactionMap.put(transactionManager, transactionStatus);pairStack.push(transactionMap);}THREAD_LOCAL.set(pairStack);}/*** 后置增强,相当于AfterReturningAdvice,方法退出时执行** 提交事务*/@AfterReturning("pointcut()")public void afterReturning() {// ※栈顶弹出(后进先出)Stack> pairStack = THREAD_LOCAL.get();while (!pairStack.empty()) {Map pair = pairStack.pop();pair.forEach((key,value)->key.commit(value));}THREAD_LOCAL.remove();}/*** 异常抛出增强,相当于ThrowsAdvice** 回滚事务*/@AfterThrowing(value = "pointcut()")public void afterThrowing() {// ※栈顶弹出(后进先出)Stack> pairStack = THREAD_LOCAL.get();logger.info("=========================");logger.info("Pair Stack:{}", pairStack);logger.info("=========================");while (!pairStack.empty()) {Map pair = pairStack.pop();pair.forEach((key,value)->key.rollback(value));}THREAD_LOCAL.remove();}
}

在事务的方法上添加前面我们的多数据源事务注解

 /*** 测试多数据源事务*/@MultiDataSourceTransactional(transactionManagers = {"masterTransactionManager", "clusterTransactionManager"})@Overridepublic void saveUser() {String rid = UUID.randomUUID().toString();User user = new User();user.setId(rid);userMapper.saveUser(user);Log log = new Log();log.setId(rid);logMapper.saveLog(log);throw new RuntimeException();}

相关内容

热门资讯

斗金订购APP贵金属期货投资被...   斗金订购APP的投资者被广告宣传给诱导,注册就送什么现金,然后充值返现金卷等等这些宣传方式,都是...
哈易购APP非法期货交易欺骗投...   哈易购APP宣传可做白银铂金贵金属订购交易,但实际上并没有取得相关交易资质!哈易购APP本质上就...
消息称百度旗下昆仑芯瞄准500... 6 月 29 日消息,据《The Information》昨日援引知情人士消息,百度旗下 AI 芯片...
打造夏日消费新场景 第35届北... 北京商报讯(记者 翟枫瑞)6月29日消息,第35届北京国际燕京啤酒文化节新闻发布会在京举行。本届啤酒...
社保基金持仓数据出炉,一季度增... 最近各大上市公司一季度财报都公开了,咱们国家社保基金的持仓数据也全部曝光。目前社保拿着比亚迪价值44...
36氪首发 | 海思、中兴团队... 作者 | 乔钰杰 编辑 | 袁斯来 硬氪获悉,广州宸思通讯科技有限公司(以下简称“宸思科技”)近日完...
两天蒸发47亿市值!一纸税务通... 一纸税务通知书,能让一家百亿龙头两天蒸发47亿市值。 6月22日,北大荒(600598.SH)公告称...
SK海力士将投资1100万亿韩... SK集团会长崔泰源6月29日在韩国“三大重大计划”发布会上宣布,公司将投资1100万亿韩元扩大半导体...
两只A股,终止上市! 两家A股公司,即将摘牌。 6月29日,退市沪科(600608.SH)公告称,上海证券交易所将在202...
原创 M... 一家成立近十年的自动驾驶公司,在IPO时吸引了14家基石投资者认购近一半的发行股份,其中不乏奔驰、比...
基金忠言|国寿安保滤镜碎,三年... 图片来源:视觉中国 蓝鲸新闻6月29日讯(记者 祁和忠)保险系基金公司国寿安保总经理换人了。 6月2...
三星电机计划加码玻璃基板!相关... 6月29日,玻璃基板概念股午后有所回升, 华工科技(000988.SZ)逼近涨停, 彩虹股份(600...
拉萨海关持续壮大外贸经营主体 ...   新华网拉萨6月28日电(记者蒋梦辰)近日,记者从拉萨海关获悉,今年前5个月,西藏有进出口实绩的外...
机构:二季报临近,医药生物板块... 6月29日,华源证券发布了一篇医药生物行业的研究报告,报告指出,业绩期临近,产业链景气度有望再次迎来...
每日收评科创50放量涨超4.5... 财联社6月29日讯,三大指数全线收红,创业板指探底回升,科创50指数大涨4.61%。沪深两市成交额3...
6月多地土拍结构性升温:深圳单... 进入2026年6月,不少城市核心区地块集中诞生高溢价宗地,热度突出的城市包含深圳、杭州、长沙。 其中...
业绩炸裂!盛达资源半年预盈3.... 6月29日,贵金属矿山龙头盛达资源(000603.SZ)发布 2026 年半年度业绩预告,上半年业绩...
A股午后拉升三大股指收涨:半导... A股三大股指6月29日开盘涨跌互现。早盘沪强深弱,创指一度跌超2%。半导体午后拉升,带动两市上涨,沪...
原创 空... 前言 大家好,我是老金。 这几天,两幅极度割裂的画面放在一起,把我看笑了。 一边是在持续的热浪下,欧...