全网整合营销服务商

电脑端+手机端+微信端=数据同步管理

免费咨询热线:400-708-3566

详解Spring AOP 实现主从读写分离

深刻讨论为什么要读写分离?

为了服务器承载更多的用户?提升了网站的响应速度?分摊数据库服务器的压力?就是为了双机热备又不想浪费备份服务器?上面这些回答,我认为都不是错误的,但也都不是完全正确的。「读写分离」并不是多么神奇的东西,也带不来多么大的性能提升,也许更多的作用的就是数据安全的备份吧。

从一个库到读写分离,从理论上对服务器压力来说是会带来一倍的性能提升,但你仔细思考一下,你的应用服务器真的很需要这一倍的提升么?那倒不如你去试着在服务器使用一下缓存系统,如 Memcached、Redis 这些分布式缓存,那性能可能是几十倍的提升。而且,在服务器硬件异常强悍及性能廉价的今天,完全更没必要了,所以,在今天,我认为它更多的职责就是为了数据安全而设计的,同时又提升了一些性能,这样也挺好。

可能我们更应该称之为主从分离

利用 AOP 实现读写分离

读写分离方式很简单,就是在你读数据是去连接从库,在你写数据的时候去连接主库,具体代码实现当然就是连接时候去操作了,这没什么难度,在代码里写就是了。可是,有追求的程序猿都是不是这么解决问题的呢!

 其实通过上篇的 Spring AOP 拦截器的基本实现 我们知道 AOP 可以实现在方法开始执行前后插入执行我们想要的代码,那这样,我们是不是可以在执行数据库操作前根据业务来动态切换数据源呢?

思考一下这个方式理论上好像是可行的,这种方式首先不需要在业务代码中去做切换,二是可能以后我们不需要读写分离了,把 AOP 切换的代码去掉就行了,三是可能就是拓展性好了。

等不了了,开始撸代码

你可能想深入的了解的话,我这里给你几个程序里用到的关键字enum(枚举)annotation(自定义注解)JoinPoint(注入点)AbstractRoutingDataSource(数据源接口子类),你理解了这些就知道了,其实你并不需要深入某些深层的东西,了解下即可。

一、建立JdbcContextHolder.java类

public class JdbcContextHolder {

 private static final ThreadLocal<String> contextHolder = new ThreadLocal<String>();

 public static void setJdbcType(String jdbcType) {
 contextHolder.set(jdbcType);
 }

 public static void setSlave() {
 setJdbcType("slave");
 }

 public static void setMaster() {
 clearJdbcType();
 }

 public static String getJdbcType() {
 return (String) contextHolder.get();
 }

 public static void clearJdbcType() {
 contextHolder.remove();
 }
}

这个类的作用就是用来设置、获取数据源连接

二、新建DynamicDataSource.java类,继承于AbstractRoutingDataSource

import org.springframework.jdbc.datasource.lookup.AbstractRoutingDataSource;

import cn.mayongfa.common.JdbcContextHolder;

public class DynamicDataSource extends AbstractRoutingDataSource {

 @Override
 protected Object determineCurrentLookupKey() {
 // 获取当前数据源连接
 return JdbcContextHolder.getJdbcType();
 }
}

通过研究,我们知道determineCurrentLookupKey方法是获取相关数据源连接的,所以重写determineCurrentLookupKey方法就可以啦,然后我们去通过刚刚我们建立的JdbcContextHolder类去获取。那怎么设置呢?

三、建立数据源DataSourceType.java枚举类

public enum DataSourceType {

 //主库
 Master("master"),

 //从库
 Slave("slave");

 private DataSourceType(String name) {
 this.name = name;
 }

 private String name;

 public String getName() {
 return name;
 }
 public void setName(String name) {
 this.name = name;
 }
}

这个枚举类的作用其实就是为了设置数据源而生的,它的目的就是让设置数据源时更方便,如丝般顺滑。

四、新建DataSource.java Annotation(自定义注解)类

import java.lang.annotation.Documented;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

@Retention(RetentionPolicy.RUNTIME) 
@Target(ElementType.METHOD)
@Documented
public @interface DataSource { 

 DataSourceType value() default DataSourceType.Master;

} 

自定义注解的意义不再过多讨论,一句话来说就是可以让你在类或方法名上以打标签的形式让该方法变得不一样。具体怎么「不一样」,这个在于你。

五、新建DataSourceChoose.java数据库切换类

import java.lang.reflect.Method;

import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.reflect.MethodSignature;

import cn.mayongfa.common.JdbcContextHolder;

public class DataSourceChoose {

//方法执行前
public void before(JoinPoint point){
 Object target = point.getTarget(); 
 String method = point.getSignature().getName(); 
 Class<?>[] classz = target.getClass().getInterfaces(); 
 MethodSignature methodSignature = (MethodSignature)point.getSignature();
 Class<?>[] parameterTypes = methodSignature.getMethod().getParameterTypes();
 try {
  Method m = classz[0].getMethod(method, parameterTypes); 
  if (m!=null && m.isAnnotationPresent(DataSource.class)) { 
  DataSource data = m.getAnnotation(DataSource.class); 
  JdbcContextHolder.clearJdbcType();
  JdbcContextHolder.setJdbcType(data.value().getName());
  } 
 } catch (Exception e) { 
  // TODO: handle exception 
 } 
}
}

这个其实是一个拦截器类,主要作用就是拦截那些方法名上有@DataSource这个自定义注解的,完了根据获取注解的value()值,来做相应的数据源切换。

到这里,整个读写分离的分析及业务逻辑和具体代码都完了,以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。


# spring  # aop  # 读写分离 


相关文章: 建站主机如何选?高性价比方案全解析  移民网站制作流程,怎么看加拿大移民官网?  测试制作网站有哪些,测试性取向的权威测试或者网站?  *服务器网站为何频现安全漏洞?  佛山网站制作系统,佛山企业变更地址网上办理步骤?  建站之星如何快速生成多端适配网站?  如何选择美橙互联多站合一建站方案?  宠物网站制作html代码,有没有专门介绍宠物如何养的网站啊?  定制建站价位费用解析与套餐推荐全攻略  在线制作视频网站免费,都有哪些好的动漫网站?  建站主机SSH密钥生成步骤及常见问题解答?  韩国代理服务器如何选?解析IP设置技巧与跨境访问优化指南  微信网站制作公司有哪些,民生银行办理公司开户怎么在微信网页上查询进度?  建站之星如何助力企业快速打造五合一网站?  建站主机是什么?如何选择适合的建站主机?  建站之星logo尺寸如何设置最合适?  专业网站制作企业网站,如何制作一个企业网站,建设网站的基本步骤有哪些?  名字制作网站免费,所有小说网站的名字?  已有域名如何免费搭建网站?  如何在宝塔面板创建新站点?  盘锦网站制作公司,盘锦大洼有多少5G网站?  娃派WAP自助建站:免费模板+移动优化,快速打造专业网站  如何获取上海专业网站定制建站电话?  建站之星价格显示格式升级,你的预算足够吗?  Python路径拼接规范_跨平台处理说明【指导】  制作网站的基本流程,设计网站的软件是什么?  专业企业网站设计制作公司,如何理解商贸企业的统一配送和分销网络建设?  制作网站公司那家好,网络公司是做什么的?  网站制作和推广的区别,想自己建立一个网站做推广,有什么快捷方法马上做好一个网站?  Android自定义listview布局实现上拉加载下拉刷新功能  香港服务器网站生成指南:免费资源整合与高速稳定配置方案  如何挑选优质建站一级代理提升网站排名?  ui设计制作网站有哪些,手机UI设计网址吗?  黑客如何通过漏洞一步步攻陷网站服务器?  如何快速搭建二级域名独立网站?  历史网站制作软件,华为如何找回被删除的网站?  如何在橙子建站中快速调整背景颜色?  已有域名和空间如何快速搭建网站?  洛阳网站制作公司有哪些,洛阳的招聘网站都有哪些?  如何在服务器上三步完成建站并提升流量?  深圳 网站制作,深圳招聘网站哪个比较好一点啊?  如何在建站主机中优化服务器配置?  建站之星安装路径如何正确选择及配置?  rsync同步时出现rsync: failed to set times on “xxxx”: Operation not permitted  招贴海报怎么做,什么是海报招贴?  济南企业网站制作公司,济南社保单位网上缴费步骤?  如何高效搭建专业期货交易平台网站?  微课制作网站有哪些,微课网怎么进?  如何快速生成橙子建站落地页链接?  网站制作软件有哪些,制图软件有哪些? 

您的项目需求

*请认真填写需求信息,我们会在24小时内与您取得联系。