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

相关内容

热门资讯

什么情况?白银突然暴涨7%逼近... 贵金属市场本周开局表现强劲。尽管围绕美伊和平谈判的最新进展再度受挫,白银价格仍升至两个月高位。 现货...
芯原股份20cm涨停,寒武纪涨... 半导体板块全线走强。芯原股份20cm涨停,寒武纪涨超17%,科创人工智能ETF易方达、科创人工智能E...
现金、动销与未来:五粮液的转身... 2026年4月30日,年报最后截止日,五粮液一纸会计差错更正公告,将2025前三季度营收从609.4...
动荡中的“压舱石”:顶级豪宅为... 文/乐居财经 严明会 “我们梳理了九大‘不确定因素’场景。虽然它们不在基准预测之列,但任何一个若兑现...
AI“三剑客”压阵!小摩:下半... 自2025年以来,新兴市场股市相对发达市场的超额收益已达25%。 这可能仅仅是开始。摩根大通认为,本...
【IPO追踪】胜宏科技(024... 5月11日,AI PCB龙头胜宏科技(02476.HK)大涨13.67%创上市以来新高,市值一举突破...
一周融资汇总:热度不减,11家... 上周(5.5-5.11)机器人行业持续迎来资本热潮。《智能新观察》基于公开信息的不完全统计,梳理出5...
原创 股... 股息到账的喜悦还未褪去,手机突然弹出一条银行扣款短信——“红利差异税扣缴xxx元”。不少股民都经历过...
注意!“三类情形”不合规发票不... “三类情形”不合规发票不能报销,这些风险点要避开! 不符合规定的发票不可以作为报销凭证,任何单位和个...
4月份CPI同比上涨1.2% 5月11日,河北石家庄,顾客在一超市内购买蔬菜。5月11日,国家统计局发布数据显示,4月份,受国际原...
轻舟智航CEO于骞:有智驾的车... 【CNMO科技消息】近日,轻舟智航联合创始人、董事长兼CEO于骞在与凤凰网财经《发现新势力》对话时,...
“双十”增长开局!宁波银行20... 近日,随着宁波银行2026年一季报及2025年年报的相继披露,这家城商行“领头羊”展现出强劲的发展韧...
原创 火... 斑马消费 范建 火锅主业增长触顶,影响资本市场信心。海底捞将破局筹码,押在了多品牌孵化之上。 202...
原创 夯... 作者|娅沁 声明|题图来源于网络。惊蛰研究所原创文章,如需转载请留言申请开白。 近两年,年轻人中开始...
美伊谈判再挫金价,市场转向交易... 据央视新闻,当地时间5月10日,美国总统特朗普在社交媒体表示,伊朗方面的回应“完全不可接受”。据新华...
宗馥莉罢免销售负责人 图片拍摄:界面新闻 赵晓娟 界面新闻记者 |赵晓娟 界面新闻编辑 |牙韩翔 娃哈哈和宏胜饮料...
直击茅台业绩说明会!回应营收确... 【导读】贵州茅台5月11日召开业绩说明会 中国基金报记者 郑俊婷 5月11日下午,贵州茅台在线上召开...
大跌41.8% 智能音箱市场遇... 快科技5月11日消息,最新行业数据显示,2026年第一季度国内智能音箱线上市场行情很冷,整体销量直接...
贵州茅台业绩会直面营利波动,王... 茅台直面了外界关注的诸多核心问题。 图片来源:贵州茅台官微 5月11日,贵州茅台酒股份有限公司(6...
2026合肥贷款中介深度评测:... 合肥专业贷款中介深度评测:合规选品,融资成功率提升65% #### 合肥贷款中介行业格局与核心挑战...