【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();}

相关内容

热门资讯

资源板块成“香饽饽”!成交额占... 财联社1月28日讯(编辑 梓隆),今日(1月28日),资源类板块成为市场关注焦点,其中,有色金属、石...
我国公募基金总规模逼近38万亿 1月28日,中基协发布的公募基金市场数据显示,截至2025年12月底,我国境内公募基金管理机构共16...
二度折戟!阳光诺和12亿关联收... (图片来源:视觉中国) 蓝鲸新闻1月28日讯(记者 邵雨婷)1月27日晚间,阳光诺和(688621....
泰国新出黄金交易限制措施 泰国皇家公报公布的新规规定,过去5年间年均黄金交易额达到或超过100亿泰铢(约合3.23亿美元)的黄...
SpaceX估值1.5万亿背后... 当特斯拉股价还在震荡时,埃隆·马斯克又给资本市场投下一枚"星际炸弹"——SpaceX计划于6月IPO...
首台中国车来了!雷军宣布小米S... 快科技1月28日消息,今日,小米CEO雷军宣布,小米SU7 Ultra将于1月29日14:00正式入...
2026年“苏超”赞助商激增:... 每经记者|王紫薇 每经编辑|文多 如果说2025年“苏超”(江苏省城市足球联赛)的赞助商们是看到流...
北大国发院最新报告:促消费的关... 今年是“十五五”规划的开局之年,中国经济版图上,“内需”二字的分量从未如此之重。 从2024年中央经...
中瑞世联资产评估集团被出具警示... 蓝鲸新闻1月28日讯,近日,湖南证监局发布行政监管措施决定书,剑指中瑞世联资产评估集团有限公司、黄新...
龙虎榜 | 8.34亿重仓!资... 1月28日,沪指涨0.27%,深证成指涨0.09%,创业板指跌0.57%。全市场成交额2.99万亿元...
原创 联... 2026年1月27日,印度和欧盟终于敲定了一份谈了将近二十年的自由贸易协议。 这一步一走,直接把全球...
多地推出以旧换新等举措激发“春... 本报记者 张芗逸 随着春节这一传统消费旺季的临近,各地纷纷加大促消费、惠民生政策力度,通过发放消费补...
万科多只债券涨幅显著 “21万... 观点网讯:1月28日 ,交易所债券市场收盘,万科多只债券表现强势。“21万科04”涨超10%,“22...
朱少醒重仓入场!黄金股的盛宴是... 牛犇 1月28日,A股继续震荡冲高,黄金相关概念股再次掀起涨停潮。那么,现在黄金股的疯狂是仍会继续还...
寿光市绿田国际商贸有限公司:专... 在农业种植与园艺培育领域,优质基质的选择直接影响作物的生长质量与产量。作为行业深耕多年的专业供应商,...
餐饮店小红书养号方法——428... 大家好,我是4288养号盒子,提供专业免费养号软件,不仅有抖音养号,还有小红书养号等、还有短视频热门...
白银狂奔,被AI“重新定价”的... 文 | 半导体产业纵横 最近,白银一路狂奔。 1月26日,伦敦现货白银盘中一度突破110美元/盎司...
6.97亿人次、↑14.2% ... 央视网消息:国家移民管理局1月28日发布的数据显示,2025年全国共6.97亿人次出入境,同比上升1...
行业领军品牌!瑞尔特以AI健康... 1月21日,由数央网、数央公益联合国内多家大众及财经媒体共同举办的2025中国消费创新大会暨第四届国...
广州:春节前后发5千万餐饮消费... 1月28日,“广货行天下 年味最广州”新闻发布会在广州举办。南都记者获悉,广州将在春节前后分多轮发放...