(本篇博客已于2019-08-28优化更新)mysql
序言:这个读写分离摘自于Springboot多数据源快速启动器,读写分离的话并非按照传统方式的springmvc看方法命名那样,而是在每一个方法上加注解或者类上添加注解代表选用的数据源。支持事物!spring
勾选Web,Mysql,MyBatis三个模块,最终建立springboot项目成功sql
<!-- springboot-aop包,AOP切面注解,Aspectd等相关注解 --> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-aop</artifactId> </dependency> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-configuration-processor</artifactId> <optional>true</optional> </dependency> <dependency> <groupId>com.baomidou</groupId> <artifactId>dynamic-datasource-spring-boot-starter</artifactId> <version>1.0.0</version> </dependency> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-jdbc</artifactId> </dependency>
server.port=8088 #数据源1 spring.datasource.db1.jdbc-url=jdbc:mysql://127.0.0.1:3306/test_msg2?useUnicode=true&characterEncoding=utf-8&useSSL=false spring.datasource.db1.username=root spring.datasource.db1.password=root spring.datasource.db1.driver-class-name=com.mysql.jdbc.Driver spring.datasource.db1.max-idle=10 spring.datasource.db1.max-wait=10000 spring.datasource.db1.min-idle=5 spring.datasource.db1.initial-size=5 #数据源2 spring.datasource.db2.jdbc-url=jdbc:mysql://127.0.0.1:3306/test_msg3?useUnicode=true&characterEncoding=utf-8&useSSL=false spring.datasource.db2.username=root spring.datasource.db2.password=root spring.datasource.db2.driver-class-name=com.mysql.jdbc.Driver spring.datasource.db2.max-idle=10 spring.datasource.db2.max-wait=10000 spring.datasource.db2.min-idle=5 spring.datasource.db2.initial-size=5 #mybatis mybatis.mapper-locations=classpath*:mapper/*.xml mybatis.type-aliases-package=com.springboot2.mjt04.dao
public class User { private int id; private String name; private int age; public User(){} public User(String name, int age){ this.name = name; this.age = age; } public int getId() { return id; } public void setId(int id) { this.id = id; } public String getName() { return name; } public void setName(String name) { this.name = name; } public int getAge() { return age; } public void setAge(int age) { this.age = age; } }
<?xml version="1.0" encoding="UTF-8" ?> <!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd" > <mapper namespace="com.springboot2.mjt04.dao.UserDao"> <!-- 查询全部user --> <select id="getAllUser" resultType="java.util.Map"> select * from user </select> <!-- 添加数据并返回主键ID,接收主键,必须以实体类接收 --> <!-- keyColumn="id"对应数据库字段,keyProperty="id"对应实体类属性 --> <insert id="addUserGetID" parameterType="com.springboot2.mjt04.entity.User" useGeneratedKeys="true" keyColumn="id" keyProperty="id"> insert into user(name,age) values(#{name},#{age}) </insert> <insert id="addUser"> insert into user(name,age) values(#{name},#{age}) </insert> </mapper>
@Mapper public interface UserDao { //使用xml配置形式查询 List<Map> getAllUser(); Long addUserGetID(User user); void addUser(User user); }
@Service public class UserService { @Autowired private UserDao moredataDao; //使用数据源1查询 @DS("datasource1") public List<Map> getAllUser1(){ return moredataDao.getAllUser(); } //使用数据源2查询 @DS("datasource2") public List<Map> getAllUser2(){ return moredataDao.getAllUser(); } //使用数据源1插入数据 @DS("datasource1") public Long addUserGetID1(User user){ return moredataDao.addUserGetID(user); } //使用数据源1插入数据 @DS("datasource2") public Long addUserGetID2(User user){ return moredataDao.addUserGetID(user); } //使用数据源1插入数据 @DS("datasource1") public void addUser1(User user){ moredataDao.addUser(user); } //使用数据源2插入数据 @DS("datasource2") public void addUser2(User user){ moredataDao.addUser(user); } @Transactional public void test1(){ moredataDao.addUserGetID(new User("mdx1",18)); int a =10/0; moredataDao.addUserGetID(new User("mdxl1",20)); } }
@Controller public class RWController { @Autowired UserService userService; //走数据源1库 @GetMapping(value = "/test1") @ResponseBody public String testOne(){ User user = new User("mjt01",20); userService.addUser1(user); return "success"; } //使用数据源2插入数据 @GetMapping(value = "/test2") @ResponseBody public String testTwo(){ User user = new User("mjt02",20); userService.addUser2(user); return "success"; } }
5个配置文件,切记若是照着写的话,当心里面的路径要准确对应上数据库
//import org.springframework.boot.autoconfigure.jdbc.DataSourceBuilder; import org.springframework.boot.context.properties.ConfigurationProperties; import org.springframework.boot.jdbc.DataSourceBuilder; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import org.springframework.context.annotation.Primary; import org.springframework.jdbc.datasource.DataSourceTransactionManager; import org.springframework.transaction.PlatformTransactionManager; import javax.sql.DataSource; import java.util.HashMap; import java.util.Map; /** * 多数据源配置类 * Created by pure on 2018-05-06. */ @Configuration public class DataSourceConfig { //数据源1 @Bean(name = "datasource1") @ConfigurationProperties(prefix = "spring.datasource.db1") // application.properteis中对应属性的前缀 public DataSource dataSource1() { return DataSourceBuilder.create().build(); } //数据源2 @Bean(name = "datasource2") @ConfigurationProperties(prefix = "spring.datasource.db2") // application.properteis中对应属性的前缀 public DataSource dataSource2() { return DataSourceBuilder.create().build(); } /** * 动态数据源: 经过AOP在不一样数据源之间动态切换 * @return */ @Primary @Bean(name = "dynamicDataSource") public DataSource dynamicDataSource() { DynamicDataSource dynamicDataSource = new DynamicDataSource(); // 默认数据源 dynamicDataSource.setDefaultTargetDataSource(dataSource1()); // 配置多数据源 Map<Object, Object> dsMap = new HashMap(); dsMap.put("datasource1", dataSource1()); dsMap.put("datasource2", dataSource2()); dynamicDataSource.setTargetDataSources(dsMap); return dynamicDataSource; } /** * 配置@Transactional注解事物 * @return */ @Bean public PlatformTransactionManager transactionManager() { return new DataSourceTransactionManager(dynamicDataSource()); } }
public class DataSourceContextHolder { /** * 默认数据源 */ public static final String DEFAULT_DS = "datasource1"; private static final ThreadLocal<String> contextHolder = new ThreadLocal<>(); // 设置数据源名 public static void setDB(String dbType) { System.out.println("切换到{"+dbType+"}数据源"); contextHolder.set(dbType); } // 获取数据源名 public static String getDB() { return (contextHolder.get()); } // 清除数据源名 public static void clearDB() { contextHolder.remove(); } }
@Retention(RetentionPolicy.RUNTIME) @Target({ElementType.METHOD}) public @interface DS { String value() default "datasource1"; }
public class DynamicDataSource extends AbstractRoutingDataSource { @Override protected Object determineCurrentLookupKey() { System.out.println("数据源为"+DataSourceContextHolder.getDB()); return DataSourceContextHolder.getDB(); } }
@Aspect @Component public class DynamicDataSourceAspect { @Before("@annotation(com.springboot2.mjt04.config.mysql.DS)") public void beforeSwitchDS(JoinPoint point){ //得到当前访问的class Class<?> className = point.getTarget().getClass(); //得到访问的方法名 String methodName = point.getSignature().getName(); //获得方法的参数的类型 Class[] argClass = ((MethodSignature)point.getSignature()).getParameterTypes(); String dataSource = DataSourceContextHolder.DEFAULT_DS; try { // 获得访问的方法对象 Method method = className.getMethod(methodName, argClass); // 判断是否存在@DS注解 if (method.isAnnotationPresent(DS.class)) { DS annotation = method.getAnnotation(DS.class); // 取出注解中的数据源名 dataSource = annotation.value(); } } catch (Exception e) { e.printStackTrace(); } // 切换数据源 DataSourceContextHolder.setDB(dataSource); } @After("@annotation(com.springboot2.mjt04.config.mysql.DS)") public void afterSwitchDS(JoinPoint point){ DataSourceContextHolder.clearDB(); } }
最终目录结构图:springboot
10.1 出现这个错误 jdbcUrl is required with driverClassName.运行出现这个过程是由于在多数据源的状况下,springboot2.0与1.X版本都不一样配置属性,解决jdbcUrl is required with driverClassName.问题,点击这里,可是我如今已经改过来了,之前是mybatis
spring.datasource.db1.url=jdbc:mysql://127.0.0.1:3306/test_msg2?useUnicode=true&characterEncoding=utf-8&useSSL=false
而如今添加了jdbc-urlmvc
spring.datasource.db1.jdbc-url=jdbc:mysql://127.0.0.1:3306/test_msg2?useUnicode=true&characterEncoding=utf-8&useSSL=false
10.2 包名路径跟换app
之前1.X版本为ide
import org.springframework.boot.autoconfigure.jdbc.DataSourceBuilder;
而如今2.X版本
import org.springframework.boot.jdbc.DataSourceBuilder;
注意一下就好,由于我都给你们改好了。
5.3 事物能够正常使用,也支持读写分离,不过在事物的方法可能比springmvc的项目麻烦些,毕竟目前个人技术只能用注解才能切换数据源,目前我没有成功找到在配置类根据方法前缀名就能够判断数据源的。
5.4 这个是application启动类上面的一个注解,代表是不使用DataSourceAutoCOnfiguration.class,由于咱们的是多个数据源,不能默认使用一个单数据源,因此要加上这个。
@SpringBootApplication(exclude = {DataSourceAutoConfiguration.class})
成功!不懂能够私信我或者留言就好。