在数据库层面大都采用读写分离技术,就是一个Master数据库,多个Slave数据库。Master库负责数据更新和实时数据查询,Slave库当然负责非实时数据查询。因为在实际的应用中,数据库都是读多写少(读取数据的频率高,更新数据的频率相对较少),而读取数据通常耗时比较长,占用数据库服务器的CPU较多,从而影响用户体验。我们通常的做法就是把查询从主库中抽取出来,采用多个从库,使用负载均衡,减轻每个从库的查询压力。

废话不多说,多数据源配置和主从数据配置原理一样
1、首先配置 jdbc.properties 两个数据库 A 和 B
#============ 双数据源 ======# #----------------------A servers--------------------------# A.driver=com.mysql.jdbc.Driver A.url=jdbc:mysql://localhost:3619/gps4?useUnicode=true&characterEncoding=utf8 A.username=gpsadmin A.password=1qaz&619 #----------------------B servers--------------------------# B.driver=com.mysql.jdbc.Driver B.url=jdbc:mysql://localhost:3619/gps6?useUnicode=true&characterEncoding=utf8 B.username=gpsadmin B.password=1qaz&619
2、配置 spring-mybatis.xml 文件【重要】
<!-- 引入配置文件 -->
<bean id="propertyConfigurer"
class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer">
<property name="location" value="classpath:resources/jdbc.properties" />
</bean>
<!-- DBCP连接池 -->
<!-- <bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource"
destroy-method="close"> <property name="driverClassName" value="${driver}"
/> <property name="url" value="${url}" /> <property name="username" value="${username}"
/> <property name="password" value="${password}" /> 初始化连接大小 <property name="initialSize"
value="${initialSize}"></property> 连接池最大数量 <property name="maxActive" value="${maxActive}"></property>
连接池最大空闲 <property name="maxIdle" value="${maxIdle}"></property> 连接池最小空闲 <property
name="minIdle" value="${minIdle}"></property> 获取连接最大等待时间 <property name="maxWait"
value="${maxWait}"></property> </``> -->
<!-- 【重点】 A 数据源 -->
<bean name="dataSourceA" class="org.springframework.jdbc.datasource.DriverManagerDataSource">
<property name="driverClassName" value="${A.driver}" />
<property name="url" value="${A.url}" />
<property name="username" value="${A.username}" />
<property name="password" value="${A.password}" />
</bean>
<!-- 【重点】 B 数据源 -->
<bean name="dataSourceB" class="org.springframework.jdbc.datasource.DriverManagerDataSource">
<property name="driverClassName" value="${B.driver}" />
<property name="url" value="${B.url}" />
<property name="username" value="${B.username}" />
<property name="password" value="${B.password}" />
</bean>
<!--【重点】 双数据源 配合 -->
<bean id="dataSource" class="com.ifengSearch.common.database.DynamicDataSource">
<property name="defaultTargetDataSource" ref="dataSourceB"/>
<property name="targetDataSources">
<map>
<entry key="dataSourceA" value-ref="dataSourceA"/>
<entry key="dataSourceB" value-ref="dataSourceB"/>
</map>
</property>
</bean>
<!-- 【重点】 加入 aop 自动扫描 DataSourceAspect 配置数据库注解aop -->
<aop:aspectj-autoproxy></aop:aspectj-autoproxy>
<bean id="manyDataSourceAspect" class="com.ifengSearch.common.database.DataSourceAspect" />
<aop:config>
<!-- 扫描 注解的 数据源 -->
<aop:aspect id="c" ref="manyDataSourceAspect">
<aop:pointcut id="tx" expression="execution(* com.ifengSearch.*.dao.*.*(..))"/>
<aop:before pointcut-ref="tx" method="before"/>
</aop:aspect>
</aop:config>
<!-- 配置数据连接 工厂 自动扫描mapping.xml文件 -->
<bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean">
<property name="dataSource" ref="dataSource"/>
<!-- 自动扫描mapping.xml文件 -->
<property name="mapperLocations" value="classpath:com/ifengSearch/*/mapping/*.xml"></property>
</bean>
<!-- DAO接口所在包名,Spring会自动查找其下的类 -->
<bean class="org.mybatis.spring.mapper.MapperScannerConfigurer">
<property name="basePackage" value="com.ifengSearch.*.dao" />
<property name="sqlSessionFactoryBeanName" value="sqlSessionFactory"></property>
</bean>
<!-- (事务管理)transaction manager, use JtaTransactionManager for global tx 事务 -->
<bean id="transactionManager"
class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource" />
</bean>
3、编写几个java类动态调用数据源【重要】
a:自定义一个注解,负责动态调用数据源
package com.ifengSearch.common.database;
import java.lang.annotation.*;
/**
* 设置 数据源 注解标签的用法 写上注解标签,
* 调用相应方法切换数据源咯(就跟你设置事务一样)
* 【也可以配置 主从数据库】
*
* @author flm
* @2017年9月12日
*/
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface DataSource {
public static String dataSourceA = "dataSourceA"; // A数据源
public static String dataSourceB = "dataSourceB"; // B数据源
String value();
}
b、数据源的获取 Object aop实现 (反射)
package com.ifengSearch.common.database;
import java.lang.reflect.Method;
import org.apache.log4j.Logger;
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.reflect.MethodSignature;
/**
* 数据源的获取
* aop实现 (反射)
* @author flm
* @2017年9月12日
*/
public class DataSourceAspect{
private Logger log = Logger.getLogger(DataSourceAspect.class);
public void before(JoinPoint point)
{
Object target = point.getTarget();// 拦截的实体类
String method = point.getSignature().getName();// 拦截的方法名称
Class<?>[] classz = target.getClass().getInterfaces();
Class<?>[] parameterTypes = ((MethodSignature) point.getSignature())
.getMethod().getParameterTypes();// 拦截的方法参数类型
try {
Method m = classz[0].getMethod(method, parameterTypes);
if (m != null && m.isAnnotationPresent(DataSource.class)) {
DataSource data = m
.getAnnotation(DataSource.class);
DataSourceHolder.setDataSource(data.value());
log.info("数据源的获取 DataSource: "+data.value());
}
} catch (Exception e) {
log.error("数据源的获取 aop实现 出错:"+e.getMessage());
}
}
}
c、DataSourceHolder 数据源操作 获取数据源 帮助类
package com.ifengSearch.common.database;
/**
* 多数据源
* 数据源操作 获取数据源
* @author flm
* @2017年9月12日
*/
public class DataSourceHolder {
//线程本地环境
private static final ThreadLocal<String> dataSources = new ThreadLocal<String>();
//设置数据源
public static void setDataSource(String customerType) {
dataSources.set(customerType);
}
//获取数据源
public static String getDataSource() {
return (String) dataSources.get();
}
//清除数据源
public static void clearDataSource() {
dataSources.remove();
}
}
d、 我们还需要实现spring的抽象类AbstractRoutingDataSource,就是实现determineCurrentLookupKey方法:
package com.ifengSearch.common.database;
import org.springframework.jdbc.datasource.lookup.AbstractRoutingDataSource;
/**
* 多数据源
* 获取数据源(依赖于spring)
* @author flm
* @2017年9月12日
*/
public class DynamicDataSource extends AbstractRoutingDataSource{
@Override
protected Object determineCurrentLookupKey() {
return DataSourceHolder.getDataSource();
}
}
4、接下来就可以看结果了
我在dao层直接调用
public interface UserDao {
/**
* 登录判断 【数据源B】
*/
@DataSource(value=DataSource.dataSourceB)
public List<UserBean> getLoginUserList(@Param("loginName")String loginName,@Param("loginPwd")String loginPwd);
/**
* 查找上一级 服务商 【数据源A】
*/
@DataSource(value=DataSource.dataSourceA)
public UserBean getServerUser(@Param("u_last_id")Integer u_last_id);
}
总结
以上所述是小编给大家介绍的Spring+Mybatis 实现aop数据库读写分离与多数据库源配置操作,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对网站的支持!
# mybatis
# 读写分离
# aop
# spring
# aop读写分离
# Spring AOP使用接口方式实现
# Springboot接口项目如何使用AOP记录日志
# Spring AOP实现Redis缓存数据库查询源码
# Spring AOP切面解决数据库读写分离实例详解
# 使用Spring AOP实现MySQL数据库读写分离案例分析(附demo)
# Spring AOP实现接口请求记录到数据库的示例代码
# 连接池
# 多个
# 新和
# 小编
# 都是
# 数据查询
# 几个
# 我在
# 在此
# 给大家
# 跟你
# 较多
# 自定义
# 还需要
# 多说
# 较少
# 较长
# 所述
# 写上
# 给我留言
相关文章:
建站主机选哪种环境更利于SEO优化?
rsync同步时出现rsync: failed to set times on “xxxx”: Operation not permitted
如何选择高效响应式自助建站源码系统?
如何基于PHP生成高效IDC网络公司建站源码?
武汉网站制作费用多少,在武汉武昌,建面100平方左右的房子,想装暖气片,费用大概是多少啊?
宝塔面板创建网站无法访问?如何快速排查修复?
如何选择CMS系统实现快速建站与SEO优化?
如何选择服务器才能高效搭建专属网站?
如何在自有机房高效搭建专业网站?
如何用西部建站助手快速创建专业网站?
网站企业制作流程,用什么语言做企业网站比较好?
网站制作公司哪里好做,成都网站制作公司哪家做得比较好,更正规?
天河区网站制作公司,广州天河区如何办理身份证?需要什么资料有预约的网站吗?
昆明网站制作哪家好,昆明公租房申请网上登录入口?
如何在建站之星绑定自定义域名?
营销式网站制作方案,销售哪个网站招聘效果最好?
如何快速生成ASP一键建站模板并优化安全性?
制作网站的基本流程,设计网站的软件是什么?
Swift中循环语句中的转移语句 break 和 continue
网站制作公司,橙子建站是合法的吗?
如何高效完成自助建站业务培训?
哈尔滨网站建设策划,哈尔滨电工证查询网站?
零基础网站服务器架设实战:轻量应用与域名解析配置指南
如何快速生成凡客建站的专业级图册?
如何高效完成独享虚拟主机建站?
如何配置支付宝与微信支付功能?
高端云建站费用究竟需要多少预算?
香港服务器如何优化才能显著提升网站加载速度?
全景视频制作网站有哪些,全景图怎么做成网页?
网站好制作吗知乎,网站开发好学吗?有什么技巧?
制作假网页,招聘网的薪资待遇,会有靠谱的吗?一面试又各种折扣?
网站制作外包价格怎么算,招聘网站上写的“外包”是什么意思?
如何高效生成建站之星成品网站源码?
SQL查询语句优化的实用方法总结
沈阳制作网站公司排名,沈阳装饰协会官方网站?
头像制作网站在线制作软件,dw网页背景图像怎么设置?
如何快速搭建高效香港服务器网站?
如何实现建站之星域名转发设置?
如何通过VPS搭建网站快速盈利?
免费视频制作网站,更新又快又好的免费电影网站?
C++中的Pimpl idiom是什么,有什么好处?(隐藏实现)
如何选择高效便捷的WAP商城建站系统?
如何设计高效校园网站?
定制建站价位费用解析与套餐推荐全攻略
如何挑选高效建站主机与优质域名?
广德云建站网站建设方案与建站流程优化指南
制作无缝贴图网站有哪些,3dmax无缝贴图怎么调?
建站主机如何选?高性价比方案全解析
如何选择最佳自助建站系统?快速指南解析优劣
常州企业建站如何选择最佳模板?
*请认真填写需求信息,我们会在24小时内与您取得联系。