springboot添加多数据源连接池并配置Mybatis
springboot添加多数据源连接池并配置Mybatis
转载请注明出处:https://www.cnblogs.com/funnyzpc/p/9190226.html
May 12, 2018 星期六,那是个晴天,天湛蓝湛蓝的非常干净,仿佛飘过一粒尘埃也能看得清清楚楚,然后就发生了些事情。。。很伤心很难过,至今也没能抹去,欸,我是怎样一步步把自己变成这个样子呢。
难过的事情总会在萦绕很久,罢了,这里还是不回忆了,就这样吧。
首先我说说这次配置多数据源的原因吧:原因大致有二:
一是我们的线上的有两大业务系统云像系统和线上交易系统,这两个系统的分别使用各自的mysql实例,交合业务的情况下目前通过定时脚本做数据更新和同步,遂在开发新的业务模块的时候就想到了将springboot配置两个数据源(mysql和mysql)
二是最近在改造数据库日志表的时候发现mysql的优化捉襟见肘,遂就想到了换数据库,换个我个人研究了许久的PostgreSQL,老大一开始不怎么乐意这么干,但是看我又研究了这么久,性能也确实较mysql高许多,再加上公司技术团队并不是很大的情况下(主要是业务量上去了数据库性能跟不上,也没有独立的DBA来维护和调优Mysql),就给了我一周的时间研究数据库(下次具体聊),故就涉及到两个数据源(mysql和PostgreSQL)的问题。
嗯,对于以上两个问题,我尝试了差异化的解决方式,对于mysl和mysql数据源我选择的是 阿里Druid+TK.Mybatis的解决方式,对于mysql和PG数据源我选择的是Hikari+TK.Mybatis的解决方式.这两种方式在实际配置中是有些许差异的,这里我略去不讲,只讲第二种,首先罗列下这其中碰到的问题:
A>DataSource和SessionFactory引用指定注入问题。
B>Hikari数据源配置参数名称差异问题。
C>Springboot init时配置类先后顺序的问题
D>多数据源下Mybatis Mapper类重复问题
我先讲讲我大致的配置过程吧,首先新建立两个配置类,以隔离开(刚在一个包 中不隔离开也可以):
然后在两个包中分别新建两个配置类,一个是MyBatis配置类和数据源、session工厂配置类,我这里是这样子:
这时候,我先展示下数据源配置类:
package **.task.config.mysql; import com.github.pagehelper.PageHelper;
import com.zaxxer.hikari.HikariConfig;
import com.zaxxer.hikari.HikariDataSource;
import org.apache.commons.lang3.StringUtils;
import org.apache.ibatis.plugin.Interceptor;
import org.apache.ibatis.session.SqlSessionFactory;
import org.mybatis.spring.SqlSessionFactoryBean;
import org.mybatis.spring.SqlSessionTemplate;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.core.io.support.PathMatchingResourcePatternResolver;
import org.springframework.core.io.support.ResourcePatternResolver;
import org.springframework.jdbc.datasource.DataSourceTransactionManager;
import org.springframework.transaction.annotation.EnableTransactionManagement; import javax.sql.DataSource;
import java.sql.SQLException;
import java.util.Properties; /**
* mybatis 配置数据源类
*
* @author
* @date 2016年12月15日
* @since 1.7
*/
@Configuration("mysqlCfg")
@EnableTransactionManagement
@ConfigurationProperties(prefix = "spring.db_mysql") //引用配置文件参数前缀
public class MybatisConfiguration extends HikariConfig {
@Value("${mybatis.mysql.xmlLocation}")
private String xmlLocation; private String typeAliasesPackage;
//配置数据源
@Bean("mysqlDataSource")
public DataSource dataSource(){
/*
HikariDataSource ds=HikariDataSource();
ds.setJdbcUrl();
ds.setConnectionTimeout();
*/
return new HikariDataSource(this);
}
//配置Session工厂
@Bean(name = "mysqlSqlSessionFactory")
public SqlSessionFactory sqlSessionFactoryBean(@Qualifier("mysqlDataSource")DataSource dataSource) {
SqlSessionFactoryBean bean = new SqlSessionFactoryBean();
bean.setDataSource(dataSource);
if(StringUtils.isNotBlank(typeAliasesPackage)){
bean.setTypeAliasesPackage(typeAliasesPackage);
}
//分页插件
PageHelper pageHelper = new PageHelper();
Properties properties = new Properties();
properties.setProperty("reasonable", "true");
properties.setProperty("supportMethodsArguments", "true");
properties.setProperty("returnPageInfo", "check");
properties.setProperty("params", "count=countSql");
pageHelper.setProperties(properties);
//添加XML目录
ResourcePatternResolver resolver = new PathMatchingResourcePatternResolver();
Interceptor[] plugins = new Interceptor[]{pageHelper};
bean.setPlugins(plugins);
try { bean.setMapperLocations(resolver.getResources(xmlLocation));
return bean.getObject();
} catch (Exception e) {
e.printStackTrace();
throw new RuntimeException(e);
}
}
//mybatis会用到的SqlSession模板
@Bean
public SqlSessionTemplate sqlSessionTemplate(@Qualifier("mysqlSqlSessionFactory") SqlSessionFactory sqlSessionFactory) {
return new SqlSessionTemplate(sqlSessionFactory);
}
//数据源事物配置
@Bean
public DataSourceTransactionManager transactionManager(@Qualifier("mysqlDataSource") DataSource dataSource) {
return new DataSourceTransactionManager(dataSource);
} }
在这里需要说明的是我通过继承HikariConfig来配置数据源以及Session工厂,配置DataSource和Session工厂的时候需要使用指定注解名称,在这里是“mysqlDataSource“和”mysqlSqlSessionFactory“,如果项目只有一个数据源的话大可不必写这个的,另外需要特别注意的是在配置session工厂一定要在形式参数前使用@Qualifier注解引用指定的数据源,同时SqlSession模板和事物也需要通过@Qualifier注解指定session工厂和数据源,这里这样做的原因是为了解决多数据源配置引用不明的问题。
OK,数据源配置完成了,现在将配置第二个配置类Mybatis配置类,以下是具体配置内容:
package **.task.config.mysql; import org.springframework.boot.autoconfigure.AutoConfigureAfter;
import org.springframework.boot.bind.RelaxedPropertyResolver;
import org.springframework.context.EnvironmentAware;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.core.env.Environment;
import tk.mybatis.spring.mapper.MapperScannerConfigurer; /**
* mybatis mapper 扫描配置类
*
* @author
* @date 2018年05月15日
*/
@Configuration("mysqlMapper")
@AutoConfigureAfter(MybatisConfiguration.class)//init时指定先后顺序,这里是数据源在mybatis之前配置
public class MapperConfiguration implements EnvironmentAware { private RelaxedPropertyResolver propertyResolver; private String basePackage;
//这里为mybatis配置指定session工厂和Dao类基础包路径
@Bean("mysqlMapperScannerConfigurer")
public MapperScannerConfigurer mapperScannerConfigurer(Environment environment){ MapperScannerConfigurer mapperScannerConfigurer = new MapperScannerConfigurer();
mapperScannerConfigurer.setSqlSessionFactoryBeanName("mysqlSqlSessionFactory");
mapperScannerConfigurer.setBasePackage(basePackage);
return mapperScannerConfigurer;
} //设置环境变量(引用配置文件中的)
@Override
public void setEnvironment(Environment environment) {
this.propertyResolver = new RelaxedPropertyResolver(environment, null);
this.basePackage = propertyResolver.getProperty("mybatis.mysql.basepackage");
}
}
配置Mybaits的时候需要将数据源配置置于之后配置,这里通过注解@AutoConfigureAfter来指定数据源配置类,在配置Mybatis引用的Session工厂时也要指定为数据源配置类中的sqlSession工厂,同时也需要指定生成的Mapper的包名,这个包的路径这里我写在application.yml的配置文件中。
配置类已经写完,现在最后一步了,在配置文件中指定配置类所引用的配置参数,大致是这样子:
spring:
application:
name: **-task
jackson:
date-format: yyyy-MM-dd HH:mm:ss
time-zone: GMT+8
default-property-inclusion: non_null
db_mysql:
name: DEV_MYSQL #LOCAL
jdbc-url: jdbc:mysql://${MYSQL_HOST:192.168.10.141}:${MYSQL_PORT:3306}/p2p?useUnicode=true&characterEncoding=UTF8
driver-class-name: com.mysql.jdbc.Driver
username: p2p
password: p2p # Hikari connection pool
type: com.zaxxer.hikari.HikariDataSource
minimum-idle: 5
maximum-pool-size: 15
auto-commit: true
idle-timeout: 30000
pool-name: DatebookHikariCP
max-lifetime: 1800000
connection-timeout: 30000
connection-test-query: SELECT 1
validation-timeout: 10000
db_pg:
name: DEV_PG # JDBC config
jdbc-url: jdbc:postgresql://192.168.10.141:5432/log
driver-class-name: org.postgresql.Driver
username: log
password: log # Hikari connection pool
type: com.zaxxer.hikari.HikariDataSource
minimum-idle: 5
maximum-pool-size: 15
auto-commit: true
idle-timeout: 30000
pool-name: DatebookHikariCP
max-lifetime: 1800000
connection-timeout: 30000
connection-test-query: SELECT 1
validation-timeout: 10000
以上配置文件中主要展示了数据源的一些配置(注意db_mysql和db_pg这两项),这里需要特别注意的是在Hikari数据源的配置参数中没有url和driverClass,只有jdbc-url 和 driver-class-name这两个,其它的配置配置参数名称与c3p0和Druid的无异,具体的连接池大小需要根据实际的项目和数据库服务器的硬件参数来配置,这里我只给出常见配置。
哦,对了,还需要在配置文件中追加Mybatis的配置参数,具体是这样:
mybatis:
mysql:
basepackage: **.task.mapper.mysql
xmlLocation: classpath:mapper/mysql/*.xml
pg:
basepackage: **.task.mapper.pg
xmlLocation: classpath:mapper/pg/*.xml
Mybatis的配置完成了,对于PostgreSQL的配置只需要注意响应的配置别名即可(比如数据源、session工厂、SqlSession工厂等等)
本节所讲的配置貌似已经完成,但是这里我顺带讲一下我在性能测试的时候所遇见的两个TK.mybatis这个插件的问题(原生mybatis也可能存在):
A>对于两个库中存在同名的Mapper名字,在@Autowired使用时会产生冲突
B>持久化需要返回主键时对于mysql和PG两中数据库的处理方式存在差异
对于以上第一个问题(多数据源Mapper冲突),我给出的解决方式是在生成的Mapper类中指定冲突的那个Mapper的Service别名,这样:
package **.task.mapper.pg; import com.github.carvechris.security.task.entity.pg.TestEmp;
import org.springframework.stereotype.Service;
import tk.mybatis.mapper.common.Mapper; @Service("pgTestEmpMapper")//这里指定别名
public interface TestEmpMapper extends Mapper<TestEmp> {
}
这个Dao在使用的时候需要使用@Resource注解来指定的Dao类:
@Autowired
@Resource(name = "pgTestEmpMapper")
private **.task.mapper.pg.TestEmpMapper pgEmpMapper;
对于以上第二个问题(持久化返回主键问题),mysql和pg的处理方式不同,具体为:
对于mysql:需要在实体类中指定主键的生成方式,即可在调用insert方法时返回生成的主键:
/**
* 表ID
*/
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)//设置为主键自增以回写主键
private Integer id;
以上的主键策略可以是主键表,也可以是UUID的方式,根据项目实际需求而定。
对于PG:首先需要在实体类的主键中这样声明:
/**
* 表ID
*/
@Id
@Column(insertable=false)//指定主键为数据库生成(同时需要在DB中将ID声明为serial类型)
private Long id;
再在Dao(生成的Mapper类中)声明一个独立的查入方法:
package **.task.mapper.pg; import **.task.entity.pg.ZwPlBalancequery;
import org.apache.ibatis.annotations.InsertProvider;
import org.a2018-06-16pache.ibatis.annotations.Options;
import org.apache.ibatis.annotations.Select;
import tk.mybatis.mapper.common.Mapper;
import tk.mybatis.mapper.provider.base.BaseInsertProvider; public interface ZwPlBalancequeryMapper extends Mapper<ZwPlBalancequery> {
//需要需要独立声明插入方法以返回插入记录的ID
@Options(useGeneratedKeys = true, keyColumn = "id", keyProperty = "id")
@InsertProvider(type = BaseInsertProvider.class, method = "dynamicSQL")
int insertWBack(ZwPlBalancequery record);
}
以上插入方法(insertWBack)中的注解是将id的生成方式改为数据库生成,至此,完美解决持久化返回记录ID问题。
现在是 2018-06-16 17:40:42 ,后天就是端午节了,预祝各位节日快乐!
springboot添加多数据源连接池并配置Mybatis的更多相关文章
- JDBC数据源连接池的配置和使用实例
个人学习参考所用,勿喷! 使用JDBC建立数据库连接的两种方式: 1.在代码中使用DriverManager获得数据库连接.这种方式效率低,并且其性能.可靠性和稳定性随着用户访问量得增加逐渐下降. 2 ...
- 【Mysql】SpringBoot阿里Druid数据源连接池配置
一.pom.xml添加 <!-- 配置数据库连接池 --> <dependency> <groupId>com.alibaba</groupId> &l ...
- Spring+Tomcat的JNDI数据源连接池简单配置
使用Tomcat JNDI数据源与Spring一起使用步骤如下: 1.将数据库驱动复制到Tomcat的lib文件夹下面 2.配置Tomcat的server.xml配置文件,在GlobalNamingR ...
- eclipse下jdbc数据源与连接池的配置及功能简介
今天在做四则运算网页版的时候遇到了一个困惑,由于需要把每个产生的式子存进 数据库,所以就需要很多次重复的加载驱动,建立连接等操作,这样一方面写程序不方便,加大了程序量,另一方面,还有导致数据库的性能急 ...
- SSM项目下Druid连接池的配置及数据源监控的使用
一,连接池的配置 在pom.xml中添加,druid的maven信息 <dependency> <groupId>com.alibaba</groupId> < ...
- Spring配置数据源(连接池)
1.数据源(连接池)的作用:为了提高程序的性能而出现的 2.数据源的原理: *事先实例化数据源,初始化部分连接资源 *使用连接资源时从数据源中获取 *使用完毕后将连接资源归还给数据源 使用c3p0的步 ...
- 配置数据库连接池,Tomcat6.0 连接池的配置
Tomcat6.0 连接池的配置1.本人当前使用的Tomcat版本为:6.0.20,oracle为稳定的9i版本 2.下文为方便起见,依习惯以%Tomcat_Home%表示Tomcat安装的目录,本人 ...
- JavaWeb之数据源连接池(1)---DBCP
何为数据源呢?也就是数据的来源.我在前面的一篇文章<JavaWeb之原生数据库连接>中,采用了mysql数据库,数据来源于mysql,那么mysql就是一种数据源.在实际工作中,除了mys ...
- 【SpringBoot笔记】SpringBoot整合Druid数据连接池
废话少说,按SpringBoot的老套路来. [step1]:添加依赖 <!-- 数据库连接池 --> <dependency> <groupId>com.alib ...
随机推荐
- Python——一个简单的进度条的实现
import math def process_bar(total_work,work_index,length): times = total_work / length # 长度倍数,用来缩放或扩 ...
- reduce/filter/map/zip/isinstance/list列表推导式
- sqlyog试用期到期--win10
1.win+R打开搜索框,输入regedit,打开windows注册表 2.删除HKEY_CURRENT_USER 下 software 的前几个随机编码.
- vue组件通信的几种方式
最近用vue开发项目,记录一下vue组件间通信几种方式 第一种,父子组件通信 一.父组件向子组件传值 1.创建子组件,在src/components/文件夹下新建一个Child.vue 2.Child ...
- python 类的魔法函数 内置函数 类方法 静态方法 抽象类
魔法函数 __init__函数 init函数会在实例化A这个类的时候被调用 class A(): def __init__(self): print('__init__函数') a = A() 显示结 ...
- day05 数据类型的方法详解
今日内容: 1.数字的基本操作 2.字符串的操作及常用方法 3.列表的操作及常用方法 重点: 1.字符串的操作及常用方法 (1)常用操作 """ 字符串的操作: &quo ...
- [Kubernetes]PV,PVC,StorageClass之间的关系详解
在Kubernetes中,容器化一个应用比较麻烦的地方莫过于对其"状态"的管理,而最常见的"状态",莫过于存储状态. 在[Kubernetes]深入理解Stat ...
- 解决sqlite 删除记录后数据库文件大小不变
最的做的项目中要有到sqlite数据存储,写了测试程序进行测试,存入300万条记录,占用flash大小为 86.1M,当把表中的记录全部删除后发后数据库文件大小依然是 86.1M: 原因是:sqlit ...
- ffmpeg推流方式采用TCP协议
ffmpeg默认推流方式采用UDP方式,若需要使用TCP协议,则需要修改. 1.使用命令时: ffmpeg 跟参数 -rtsp_transport tcp 2.编码时 AVFormatContext ...
- 【转载的】这张图能容易理解sql joins,收藏下!