当前位置: 首页 > news >正文

浏阳市商务局网站溪江农贸市场建设网站建设 深圳

浏阳市商务局网站溪江农贸市场建设,网站建设 深圳,建设网站的英语怎么说,淘宝客做网站怎么做转载自 springboot-mybatis多数据源的两种整合方法 简介#xff1a; 随着并发量的不断增加#xff0c;显然单个数据库已经承受不了高并发带来的压力。一个项目使用多个数据库#xff08;无论是主从复制- - 读写分离还是分布式数据库结构#xff09;的重要性变得越来越明显…转载自  springboot-mybatis多数据源的两种整合方法 简介 随着并发量的不断增加显然单个数据库已经承受不了高并发带来的压力。一个项目使用多个数据库无论是主从复制- - 读写分离还是分布式数据库结构的重要性变得越来越明显。传统项目中个人对传统项目的理解就是所有的业务模块都在一个tomcat中完成多个相同的tomcat集群也可认为是传统项目整合多数据源有两种方法分包和AOP。 版本 springboot1.5.9.RELEASE  mariadb5.7 一、分包方式实现 1、在application.properties中配置两个数据库 ## test1 database spring.datasource.test1.urljdbc:mysql://localhost:3307/multipledatasource1?useUnicodetruecharacterEncodingUTF-8serverTimezoneUTCuseSSLfalse spring.datasource.test1.usernameroot spring.datasource.test1.passwordroot spring.datasource.test1.driver-class-namecom.mysql.cj.jdbc.Driver ## test2 database spring.datasource.test2.urljdbc:mysql://localhost:3307/multipledatasource2?useUnicodetruecharacterEncodingUTF-8serverTimezoneUTCuseSSLfalse spring.datasource.test2.usernameroot spring.datasource.test2.passwordroot spring.datasource.test2.driver-class-namecom.mysql.cj.jdbc.Driver 2、建立连个数据源的配置文件 springbooot中的参数可以参考上一篇博客不定期更新中https://blog.csdn.net/tuesdayma/article/details/81029539 第一个配置文件 //表示这个类为一个配置类 Configuration // 配置mybatis的接口类放的地方 MapperScan(basePackages com.mzd.multipledatasources.mapper.test01, sqlSessionFactoryRef test1SqlSessionFactory) public class DataSourceConfig1 {// 将这个对象放入Spring容器中Bean(name test1DataSource)// 表示这个数据源是默认数据源Primary// 读取application.properties中的配置参数映射成为一个对象// prefix表示参数的前缀ConfigurationProperties(prefix spring.datasource.test1)public DataSource getDateSource1() {return DataSourceBuilder.create().build();}Bean(name test1SqlSessionFactory)// 表示这个数据源是默认数据源Primary// Qualifier表示查找Spring容器中名字为test1DataSource的对象public SqlSessionFactory test1SqlSessionFactory(Qualifier(test1DataSource) DataSource datasource)throws Exception {SqlSessionFactoryBean bean new SqlSessionFactoryBean();bean.setDataSource(datasource);bean.setMapperLocations(// 设置mybatis的xml所在位置new PathMatchingResourcePatternResolver().getResources(classpath*:mapping/test01/*.xml));return bean.getObject();}Bean(test1SqlSessionTemplate)// 表示这个数据源是默认数据源Primarypublic SqlSessionTemplate test1sqlsessiontemplate(Qualifier(test1SqlSessionFactory) SqlSessionFactory sessionfactory) {return new SqlSessionTemplate(sessionfactory);} } 第二个配置文件 Configuration MapperScan(basePackages com.mzd.multipledatasources.mapper.test02, sqlSessionFactoryRef test2SqlSessionFactory) public class DataSourceConfig2 {Bean(name test2DataSource)ConfigurationProperties(prefix spring.datasource.test2)public DataSource getDateSource2() {return DataSourceBuilder.create().build();}Bean(name test2SqlSessionFactory)public SqlSessionFactory test2SqlSessionFactory(Qualifier(test2DataSource) DataSource datasource)throws Exception {SqlSessionFactoryBean bean new SqlSessionFactoryBean();bean.setDataSource(datasource);bean.setMapperLocations(new PathMatchingResourcePatternResolver().getResources(classpath*:mapping/test02/*.xml));return bean.getObject();}Bean(test2SqlSessionTemplate)public SqlSessionTemplate test2sqlsessiontemplate(Qualifier(test2SqlSessionFactory) SqlSessionFactory sessionfactory) {return new SqlSessionTemplate(sessionfactory);} } 注意 1、Primary这个注解必须要加因为不加的话spring将分不清楚那个为主数据源默认数据源 2、mapper的接口、xml形式以及dao层都需要两个分开目录如图    3、bean.setMapperLocations(new PathMatchingResourcePatternResolver().getResources(“XXXX”));mapper的xml形式文件位置必须要配置不然将报错no statement 这种错误也可能是mapper的xml中namespace与项目的路径不一致导致的具体看情况吧注意一下就行问题不大的 4、在service层中根据不同的业务注入不同的dao层。 5、如果是主从复制- -读写分离比如test01中负责增删改test02中负责查询。但是需要注意的是负责增删改的数据库必须是主库master 6、如果是分布式结构的话不同模块操作各自的数据库就好test01包下全是test01业务test02全是test02业务但是如果test01中掺杂着test02的编辑操作这时候将会产生事务问题即test01中的事务是没法控制test02的事务的这个问题在之后的博客中会解决。 二、AOP实现 简介 用这种方式实现多数据源的前提必须要清楚两个知识点AOP原理和AbstractRoutingDataSource抽象类。 1、AOP这个东西。。。不切当的说就是相当于拦截器只要满足要求的都会被拦截过来然后进行一些列的操作。具体需要自己去体会。。。 2、AbstractRoutingDataSource这个类是实现多数据源的关键他的作用就是动态切换数据源实质有多少个数据源就存多少个数据源在targetDataSources是AbstractRoutingDataSource的一个map类型的属性其中value为每个数据源key表示每个数据源的名字这个属性中然后根据determineCurrentLookupKey这个方法获取当前数据源在map中的key值然后determineTargetDataSource方法中动态获取当前数据源如果当前数据源不存并且默认数据源也不存在就抛出异常。 public abstract class AbstractRoutingDataSource extends AbstractDataSource implements InitializingBean {//多数据源map集合private MapObject, Object targetDataSources;//默认数据源private Object defaultTargetDataSource;//其实就是targetDataSources后面的afterPropertiesSet方法会将targetDataSources赋值给resolvedDataSourcesprivate MapObject, DataSource resolvedDataSources;private DataSource resolvedDefaultDataSource;public void setTargetDataSources(MapObject, Object targetDataSources) {this.targetDataSources targetDataSources;}protected DataSource determineTargetDataSource() {Assert.notNull(this.resolvedDataSources, DataSource router not initialized);Object lookupKey this.determineCurrentLookupKey();DataSource dataSource (DataSource)this.resolvedDataSources.get(lookupKey);if (dataSource null (this.lenientFallback || lookupKey null)) {dataSource this.resolvedDefaultDataSource;}if (dataSource null) {throw new IllegalStateException(Cannot determine target DataSource for lookup key [ lookupKey ]);} else {return dataSource;}}protected abstract Object determineCurrentLookupKey(); } 具体实现 1、定义一个动态数据源继承AbstractRoutingDataSource 抽象类并重写determineCurrentLookupKey方法 public class DynamicDataSource extends AbstractRoutingDataSource {Overrideprotected Object determineCurrentLookupKey() {DataSourceType.DataBaseType dataBaseType DataSourceType.getDataBaseType();return dataBaseType;}} 2、创建一个切换数据源类型的类 ThreadLocal这个知识点可以参考我的博客https://blog.csdn.net/tuesdayma/article/details/74841657 就是为了线程的安全性每个线程之间不会相互影响。 public class DataSourceType {public enum DataBaseType {TEST01, TEST02}// 使用ThreadLocal保证线程安全private static final ThreadLocalDataBaseType TYPE new ThreadLocalDataBaseType();// 往当前线程里设置数据源类型public static void setDataBaseType(DataBaseType dataBaseType) {if (dataBaseType null) {throw new NullPointerException();}System.err.println([将当前数据源改为] dataBaseType);TYPE.set(dataBaseType);}// 获取数据源类型public static DataBaseType getDataBaseType() {DataBaseType dataBaseType TYPE.get() null ? DataBaseType.TEST01 : TYPE.get();System.err.println([获取当前数据源的类型为] dataBaseType);return dataBaseType;}// 清空数据类型public static void clearDataBaseType() {TYPE.remove();}} 3、定义多个数据源怎么定义就不多说了和方法一是一样的主要是将定义好的多个数据源放在动态数据源中。 Configuration MapperScan(basePackages com.mzd.multipledatasources.mapper, sqlSessionFactoryRef SqlSessionFactory) public class DataSourceConfig {PrimaryBean(name test1DataSource)ConfigurationProperties(prefix spring.datasource.test1)public DataSource getDateSource1() {return DataSourceBuilder.create().build();}Bean(name test2DataSource)ConfigurationProperties(prefix spring.datasource.test2)public DataSource getDateSource2() {return DataSourceBuilder.create().build();}Bean(name dynamicDataSource)public DynamicDataSource DataSource(Qualifier(test1DataSource) DataSource test1DataSource,Qualifier(test2DataSource) DataSource test2DataSource) {MapObject, Object targetDataSource new HashMap();targetDataSource.put(DataSourceType.DataBaseType.TEST01, test1DataSource);targetDataSource.put(DataSourceType.DataBaseType.TEST02, test2DataSource);DynamicDataSource dataSource new DynamicDataSource();dataSource.setTargetDataSources(targetDataSource);dataSource.setDefaultTargetDataSource(test1DataSource);return dataSource;}Bean(name SqlSessionFactory)public SqlSessionFactory test1SqlSessionFactory(Qualifier(dynamicDataSource) DataSource dynamicDataSource)throws Exception {SqlSessionFactoryBean bean new SqlSessionFactoryBean();bean.setDataSource(dynamicDataSource);bean.setMapperLocations(new PathMatchingResourcePatternResolver().getResources(classpath*:mapping/*.xml));return bean.getObject();} } 4、定义AOP就是不同业务切换不同数据库的入口。如果觉得execution太长不愿意写就可以定义一个注解来实现。可参考于我的博客https://blog.csdn.net/tuesdayma/article/details/79704238 Aspect Component public class DataSourceAop {Before(execution(* com.mzd.multipledatasources.service..*.test01*(..)))public void setDataSource2test01() {System.err.println(test01业务);DataSourceType.setDataBaseType(DataBaseType.TEST01);}Before(execution(* com.mzd.multipledatasources.service..*.test02*(..)))public void setDataSource2test02() {System.err.println(test02业务);DataSourceType.setDataBaseType(DataBaseType.TEST02);} } 整体目录如图  源代码https://github.com/mzd123/springboot-multipledatasources/tree/master
http://www.zqtcl.cn/news/943727/

相关文章:

  • 临沂网站设计哪家好qq浏览器网页版进入
  • seo资料站哔哩哔哩官方网站首页
  • 前端怎么做网站万网域名管理入口
  • asp.net 做网站实例特别酷炫网站
  • 个人网站的内容网页设计图片显示不出来怎么弄
  • 福建省建设人才与科技发展中心网站首页关于制作网站收费标准
  • 什么软件可以发帖子做推广中山优化网站
  • 中山网站建设开发网络营销的基本功能
  • 温州平阳县网站建设兼职免费下载简历模板
  • 导购网站 转化率wordpress 拓展
  • 美文分享网站源码互联网网站建设
  • 做网站用php还是python建设网站价格
  • 平台网站怎么做诱导网站怎么做
  • 网站建设人员构成网址申请域名
  • 微网站建设找哪家公司好郑州一凡网站建设
  • 江阴网站制作公司泉州网站建设论坛
  • 最新章节 62.一起来做网站吧时钟插件+wordpress
  • 惠州市建设规划局网站网页设计实训报告word
  • 大众汽车网站建设鳌江网站建设
  • 佛山外贸网站建设公司网站与网页区别
  • HTML网站建设课程微商怎么做网站
  • 专业数据分析网站wordpress 很差
  • 请人做个网站多少钱google推广妙招
  • 郑州销售网站开一个设计公司
  • 建筑公司网站常用长尾词网页设计实训总结100字
  • 网站开发项目业务要求wordpress前台注册登陆
  • 上海人才网官网招聘人力资源专业wordpress seo title
  • 简单html网站网页设计培训学费多少
  • 麻城网站建设投标网招标网
  • 网站建设行业细分专业动漫如何制作