spring boot 入门_SpringBoot入门建站全系列(三十)Mybatis多数据源进行数据库操作
SpringBoot入门建站全系列(三十)Mybatis多数据源进行数据库操作
一、概述
多数据源,就是有多个数据库的配置。
多数据源配置并不麻烦,使用起来和单数据源基本相同,但是,重要的是事务的控制。
这就牵扯到分布式事务控制,本篇只是讲述多数据源的使用,下一篇再讲述如何使用分布式事务控制中间件。
首发地址: 品茗IT-首发
如果大家正在寻找一个java的学习环境,或者在开发中遇到困难,可以加入我们的java学习圈,点击即可加入,共同学习,节约学习时间,减少很多在学习中遇到的难题。
二、配置
本文假设你已经引入spring-boot-starter-web。已经是个SpringBoot项目了,如果不会搭建,可以打开这篇文章看一看《SpringBoot入门建站全系列(一)项目建立》。
2.1 Maven依赖
引入mybatis、数据库连接池、mysql-connector.
<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-web</artifactId>
</dependency>
<dependency><groupId>org.mybatis.spring.boot</groupId><artifactId>mybatis-spring-boot-starter</artifactId>
</dependency>
<dependency><groupId>mysql</groupId><artifactId>mysql-connector-java</artifactId>
</dependency>
<dependency><groupId>org.apache.commons</groupId><artifactId>commons-dbcp2</artifactId>
</dependency>
2.2 配置文件
在application.properties 中需要配置数据库相关信息的信息,如:
spring.datasource.master.type=org.apache.commons.dbcp2.BasicDataSource
spring.datasource.master.max-wait-millis=60000
spring.datasource.master.min-idle=20
spring.datasource.master.initial-size=2
spring.datasource.master.validation-query=SELECT 1
spring.datasource.master.connection-properties=characterEncoding=utf8
spring.datasource.master.validation-query=SELECT 1
spring.datasource.master.test-while-idle=true
spring.datasource.master.test-on-borrow=true
spring.datasource.master.test-on-return=falsespring.datasource.master.driverClassName = com.mysql.jdbc.Driver
spring.datasource.master.url=jdbc:mysql://127.0.0.1:3306/boot?useUnicode=true&characterEncoding=utf8&serverTimezone=UTC
spring.datasource.master.username=cff
spring.datasource.master.password=123456spring.datasource.slave.type=org.apache.commons.dbcp2.BasicDataSource
spring.datasource.slave.max-wait-millis=60000
spring.datasource.slave.min-idle=20
spring.datasource.slave.initial-size=2
spring.datasource.slave.validation-query=SELECT 1
spring.datasource.slave.connection-properties=characterEncoding=utf8
spring.datasource.slave.validation-query=SELECT 1
spring.datasource.slave.test-while-idle=true
spring.datasource.slave.test-on-borrow=true
spring.datasource.slave.test-on-return=falsespring.datasource.slave.driverClassName = com.mysql.jdbc.Driver
spring.datasource.slave.url=jdbc:mysql://127.0.0.1:3306/boot?useUnicode=true&characterEncoding=utf8&serverTimezone=UTC
spring.datasource.slave.username=cff
spring.datasource.slave.password=123456mybatis.configuration.log-impl=org.apache.ibatis.logging.stdout.StdOutImpl
注意,这里的配置:
- 虽然是spring.datasource开头,但是已经不是需要Spring自动装配的配置了;
- spring.datasource.master开头的配置,是需要我手动配置master数据源的配置;
- spring.datasource.slave开头的配置,是需要我手动配置slave数据源的配置;
- master和slave只是个名字而已,根据自己需要修改任何名称都行;
- mybatis.configuration.log-impl是mybatis自动装配的配置。
三、配置数据源
配置多个数据源,需要DataSource、TransactionManager、SqlSessionFactory
3.1 Master数据源
配置Master数据源,需要Master自己的DataSource、TransactionManager、SqlSessionFactory。
同时,指定MapperScan扫描的包路径,和对应的sqlSessionFactory。
package com.cff.springbootwork.atomikos.config;import org.apache.commons.dbcp2.BasicDataSource;
import org.apache.ibatis.session.SqlSessionFactory;
import org.mybatis.spring.SqlSessionFactoryBean;
import org.mybatis.spring.annotation.MapperScan;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.Primary;
import org.springframework.jdbc.datasource.DataSourceTransactionManager;@Configuration
@MapperScan(basePackages = MasterDataSourceConfiguration.PACKAGE, sqlSessionFactoryRef = "masterSqlSessionFactory")
public class MasterDataSourceConfiguration {static final String PACKAGE = "com.cff.springbootwork.atomikos.dao.master";@Bean(name = "masterDataSource")@ConfigurationProperties(prefix = "spring.datasource.master")@Primarypublic BasicDataSource masterDataSource() {BasicDataSource dataSource = new BasicDataSource();return dataSource;}@Bean(name = "masterTransactionManager")@Primarypublic DataSourceTransactionManager masterTransactionManager() {return new DataSourceTransactionManager(masterDataSource());}@Bean(name = "masterSqlSessionFactory")@Primarypublic SqlSessionFactory masterSqlSessionFactory() throws Exception {final SqlSessionFactoryBean sessionFactory = new SqlSessionFactoryBean();sessionFactory.setDataSource(masterDataSource());return sessionFactory.getObject();}
}
这里面,
- 新建了一个BasicDataSource,bean名称加个master以做区分,BasicDataSource的属性通过@ConfigurationProperties(prefix = "spring.datasource.master")注入,同时,加上@Primary标明它是首选。
- @Primary注解只能加在某一个数据源上,不能所有数据源配置都加,不然会报错。
- 新建了一个masterTransactionManager,处理masterDataSource的事务。
- 新建了masterSqlSessionFactory,处理masterDataSource的SqlSession。
3.2 Slave数据源
配置Slave数据源,需要Slave自己的DataSource、TransactionManager、SqlSessionFactory。
同时,指定MapperScan扫描的包路径,和对应的sqlSessionFactory。
package com.cff.springbootwork.atomikos.config;import org.apache.commons.dbcp2.BasicDataSource;
import org.apache.ibatis.session.SqlSessionFactory;
import org.mybatis.spring.SqlSessionFactoryBean;
import org.mybatis.spring.annotation.MapperScan;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.jdbc.datasource.DataSourceTransactionManager;@Configuration
@MapperScan(basePackages = SlaveDataSourceConfiguration.PACKAGE, sqlSessionFactoryRef = "slaveSqlSessionFactory")
public class SlaveDataSourceConfiguration {static final String PACKAGE = "com.cff.springbootwork.atomikos.dao.slave";@Bean(name = "slaveDataSource")@ConfigurationProperties(prefix = "spring.datasource.slave")public BasicDataSource slaveDataSource() {BasicDataSource dataSource = new BasicDataSource();return dataSource;}@Bean(name = "slaveTransactionManager")public DataSourceTransactionManager slaveTransactionManager() {return new DataSourceTransactionManager(slaveDataSource());}@Bean(name = "slaveSqlSessionFactory")public SqlSessionFactory slaveSqlSessionFactory() throws Exception {final SqlSessionFactoryBean sessionFactory = new SqlSessionFactoryBean();sessionFactory.setDataSource(slaveDataSource());return sessionFactory.getObject();}
}
这里面,
- 新建了一个BasicDataSource,bean名称加个slave以做区分,BasicDataSource的属性通过@ConfigurationProperties(prefix = "spring.datasource.slave")注入,注意,这个地方不能加上@Primary注解。
- 新建了一个slaveTransactionManager,处理slaveDataSource的事务。
- 新建了slaveSqlSessionFactory,处理slaveDataSource的SqlSession。
四、测试Mapper
Mapper的写法无任何改变,只不过,要注意不同mapper放在上面不同数据源指定的包路径下。
4.1 Master数据源的Mapper
新建一个UserInfoMasterDao,处理master数据源的用户表。
UserInfoMasterDao:
package com.cff.springbootwork.atomikos.dao.master;import org.apache.ibatis.annotations.Mapper;
import org.apache.ibatis.annotations.Param;
import org.apache.ibatis.annotations.Select;
import org.apache.ibatis.annotations.Update;import com.cff.springbootwork.atomikos.domain.UserInfo;@Mapper
public interface UserInfoMasterDao {@Select({"<script>","SELECT ","user_name as userName,passwd,name,mobile,valid, user_type as userType","FROM user_info","WHERE user_name = #{userName,jdbcType=VARCHAR}","</script>"})UserInfo findByUserName(@Param("userName") String userName);@Update({"<script>"," update user_info set"," name = #{name, jdbcType=VARCHAR}, mobile = #{mobile, jdbcType=VARCHAR}"," where user_name=#{userName}","</script>"})int update(UserInfo userInfo);
}
4.2 Slave数据源的Mapper
新建一个UserInfoSlaveDao,处理Slave数据源的用户表。
UserInfoSlaveDao:
package com.cff.springbootwork.atomikos.dao.slave;import org.apache.ibatis.annotations.Mapper;
import org.apache.ibatis.annotations.Param;
import org.apache.ibatis.annotations.Select;
import org.apache.ibatis.annotations.Update;import com.cff.springbootwork.atomikos.domain.UserInfo;@Mapper
public interface UserInfoSlaveDao {@Select({"<script>","SELECT ","user_name as userName,passwd,name,mobile,valid, user_type as userType","FROM user_info","WHERE user_name = #{userName,jdbcType=VARCHAR}","</script>"})UserInfo findByUserName(@Param("userName") String userName);@Update({"<script>"," update user_info set"," name = #{name, jdbcType=VARCHAR}, mobile = #{mobile, jdbcType=VARCHAR}"," where user_name=#{userName}","</script>"})int update(UserInfo userInfo);
}
五、测试Service
写一个普通的UserInfoService,注入两个mapper即可使用。
UserInfoService:
package com.cff.springbootwork.atomikos.service;import java.util.ArrayList;
import java.util.List;import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional;import com.cff.springbootwork.atomikos.dao.master.UserInfoMasterDao;
import com.cff.springbootwork.atomikos.dao.slave.UserInfoSlaveDao;
import com.cff.springbootwork.atomikos.domain.UserInfo;@Service
public class UserInfoService {@AutowiredUserInfoMasterDao userInfoMasterDao;@AutowiredUserInfoSlaveDao userInfoSlaveDao;public List<UserInfo> getUserInfoByUserName(String userName) {UserInfo userInfoMaster = userInfoMasterDao.findByUserName(userName);UserInfo userInfoSlave = userInfoSlaveDao.findByUserName(userName);List<UserInfo> list = new ArrayList<>();list.add(userInfoMaster);list.add(userInfoSlave);return list;}@Transactional(rollbackFor = Exception.class)public List<UserInfo> update(UserInfo userInfo) {userInfoMasterDao.update(userInfo);userInfoSlaveDao.update(userInfo);int i = 1 / 0;return getUserInfoByUserName(userInfo.getUserName());}
}
注意,我这里的update方法中,写了一个int i = 1 / 0;
,是为了测试无分布式事务支持的时候事务是如何处理的。
六、测试Web
package com.cff.springbootwork.atomikos.web;import java.util.List;import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.PathVariable;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.bind.annotation.RestController;import com.cff.springbootwork.atomikos.domain.UserInfo;
import com.cff.springbootwork.atomikos.service.UserInfoService;@RestController
@RequestMapping("/mybatis")
public class MybatisRest {@AutowiredUserInfoService userInfoService;@RequestMapping(value = "/mybatis/{name}", method = { RequestMethod.GET })public List<UserInfo> testMybatis(@PathVariable("name") String name) {return userInfoService.getUserInfoByUserName(name);}@RequestMapping(value = "/update")public List<UserInfo> update(@RequestBody UserInfo userInfo) {return userInfoService.update(userInfo);}
}
七、实体
UserInfo:
完整实体,请访问《SpringBoot入门建站全系列(三十)Mybatis多数据源进行数据库操作》查看。
七、结论
- 多数据源的使用和普通使用方法完全一样,只不过要多配置一下。
- 多数据源的事务无法正常回滚,需要分布式事务控制。
- 多数据源的事务,会回滚第一个,第二个回滚不了。
品茗IT-博客专题:https://www.pomit.cn/lecture.html汇总了Spring专题、Springboot专题、SpringCloud专题、web基础配置专题。
快速构建项目
Spring项目快速开发工具:
一键快速构建Spring项目工具
一键快速构建SpringBoot项目工具
一键快速构建SpringCloud项目工具
一站式Springboot项目生成
Mysql一键生成Mybatis注解Mapper
Spring组件化构建
SpringBoot组件化构建
SpringCloud服务化构建
喜欢这篇文章么,喜欢就加入我们一起讨论SpringBoot使用吧!
spring boot 入门_SpringBoot入门建站全系列(三十)Mybatis多数据源进行数据库操作相关推荐
- springboot mybatis ehcache_SpringBoot入门建站全系列(十四)集成Redis缓存
SpringBoot入门建站全系列(十四)集成Redis缓存 一.概述 本地缓存,就是使用应用内使用本地内存将数据暂缓存储,一般数据库的查询如果不怎么改动,可以用本地缓存暂存. 远程缓存,比如redi ...
- python建站部署_SpringBoot入门建站全系列(三十二)接入xxl-job分布式任务调度平台...
SpringBoot入门建站全系列(三十二)接入xxl-job分布式任务调度平台 一.概述 XXL-JOB是一个轻量级分布式任务调度平台,其核心设计目标是开发迅速.学习简单.轻量级.易扩展.现已开放源 ...
- kafka maven 依赖_SpringBoot入门建站全系列(二十八)整合Kafka做日志监控
SpringBoot入门建站全系列(二十八)整合Kafka做日志监控 一.概述 Apache Kafka是一个分布式发布 - 订阅消息系统和一个强大的队列,可以处理大量的数据,并使您能够将消息从一个端 ...
- SpringBoot入门建站全系列(二十八)整合Kafka做日志监控
SpringBoot入门建站全系列(二十八)整合Kafka做日志监控 一.概述 Apache Kafka是一个分布式发布 - 订阅消息系统和一个强大的队列,可以处理大量的数据,并使您能够将消息从一个端 ...
- boot spring test 文档_SpringBoot入门建站全系列(五)使用Spring-data-jpa操作数据库
SpringBoot入门建站全系列(五)使用Spring-data-jpa操作数据库 SpringBoot操作数据库有多种方式,如 JDBC直接操作:太古老了,没人愿意这样玩 Mybatis插件:比较 ...
- spring配置文件_SpringBoot入门建站全系列(二十三)配置文件优先级及自定义配置文件...
SpringBoot入门建站全系列(二十三)配置文件优先级及自定义配置文件 一.概述 Spring Boot允许多种配置来源,官网是这样说的: Spring Boot使用一种非常特殊的Property ...
- springboot util 测试类怎么写_SpringBoot入门建站全系列(九)文件上传功能与下载方式...
SpringBoot入门建站全系列(九)文件上传功能与下载方式 Spring对文件上传做了简单的封装,就是用MultipartFile这个对象去接收文件,当然有很多种写法,下面会一一介绍. 文件的下载 ...
- SpringBoot入门建站全系列(二十七)WebSocket做简单的聊天室
SpringBoot入门建站全系列(二十七)WebSocket做简单的聊天室 一.概述 WebSocket 是一种网络通信协议.RFC6455 定义了它的通信标准. WebSocket 是 HTML5 ...
- SpringBoot入门建站全系列(二十六)Mongodb非关系型数据库的使用
SpringBoot入门建站全系列(二十六)Mongodb非关系型数据库的使用 一.概述 MongoDB 是一个基于分布式文件存储的数据库.由 C++ 语言编写.旨在为 WEB 应用提供可扩展的高性能 ...
最新文章
- 如果根据日志去禁用user_agent
- MATLAB从入门到精通-欧拉法与梯形法求解微分方程(含MATLAB源码)
- HDU 3336 KMP
- BugkuCTF-Misc:come_game
- [Oracle][Corruption]究竟哪些检查影响到 V$DATABASE_BLOCK_CORRUPTION
- 用IE和Google 浏览器的比较。。
- 用Python实现黑客帝国代码雨效果
- 异速联:解决打印样式个性化设置
- HTML学生个人网站作业设计——HTML+CSS+JavaScript简单的大学生书店网页制作(13页) web期末作业设计网页 web结课作业的源码 web网页设计实例作业
- 点聚AIP电子签章在Chrome、Edge新内核、360系列、FireFox、IE、Opera等浏览器中实现网页内嵌效果案例(一)
- css border流光效果
- 一款类似windows wallpaper engine的Mac动画壁纸软件
- python和工程造价_不会Python的将慢慢被淘汰!
- 【无标题】体验scratch海底世界
- DELL R510服务器RAID配置,Dell服务器硬RAID配置
- Android做3D旋转动画,Android编程实现3D旋转效果实例
- 跟开涛老师学shiro -- 授权
- echarts 盒须图配置
- 【课上笔记】第五章 数组和广义表
- 复杂网络多局域世界模型matlab算法实现
热门文章
- 无法将类型为“System.__ComObject”的 COM 对象强制转换为类类型“mshtml.HTMLInputElementClass...
- Resource interpreted as Script but transferred with MIME type text/plain:
- 重构学习2-消除过长参数
- 在使用模板时遇到的错误,TypeError at /index/ context must be a dict rather than RequestContext(亲测)
- pandas速查手册(中文版)
- 支付宝公共服务窗开发总结
- mysql使用MRG_MyISAM(MERGE)实现水平分表
- Go的marshal unmarshal
- pb 更改dropdwonlistbox绑定数据_Blazor 修仙之旅 组件与数据绑定
- 普罗米修斯java_springboot集成普罗米修斯(Prometheus)的方法