【转载】基于ssm框架进行读写分离(使用ReplicationDriver),部分修改

本文介绍了一种简单快捷的MySQL读写分离方案,通过使用ReplicationDriver和Spring配置实现读写分离,适用于springmvc+mybatis框架。文章详细介绍了配置过程及注意事项。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

这是我见过最简单快捷的读写分离方案,亲测不错!转载文章部分修改补漏。


1、先配置好MySQL proxy方式的主从复制;


2、用mysql  Connector提供的ReplicationDriver实现读写分离,这是目前最新也是比较好的读写分离方式。

我的框架是springmvc+mybatis,数据库连接池使用的是阿里的druid,他们的具体配置这里就不说了,主要说和读写分离有关的地方。

(1)要使用driver不能是之前的默认的com.mysql.jdbc.Driver,而要使用提供的com.mysql.jdbc.ReplicationDriver,并且url也不是单个的url了,详细如下:

db.connection.driver=com.mysql.jdbc.ReplicationDriver  
db.connection.url=jdbc:mysql:replication://192.168.1.99:3306,192.168.1.92:3306/testdb?useUnicode=true&characterEncoding=UTF-8&autoReconnect=false&useSSL=false&failOverReadOnly=true&loadBalanceStrategy=random&readFormMasterNoSlaves=true  
db.connection.username=root  
db.connection.password=password 

其中192.168.1.99是主从数据库中的master,92是slave,如果有多个slave可以继续在后面追加,注意数据库是放在最后面的,然后是后面的一串东西,建议最好和我这个保持一致。

spring配置的地方如下,部分代码,主要是diverClassName这个地方,默认是没有的,这里需要制定下:


<!-- 数据源 -->  
    <!--see https://github.com/alibaba/druid/wiki/%E9%85%8D%E7%BD%AE_DruidDataSource%E5%8F%82%E8%80%83%E9%85%8D%E7%BD%AE -->  
    <bean id="dataSource" class="com.alibaba.druid.pool.DruidDataSource"  
        init-method="init" destroy-method="close">  
        <!-- 基本属性 url、user、password -->  
        <property name="driverClassName" value="${db.connection.driver}" />  
        <property name="url" value="${db.connection.url}" />  
        <property name="username" value="${db.connection.username}" />  
        <property name="password" value="${db.connection.password}" />  


(2)为了实现读写分离,在下面事务配置的地方非常重要,什么get、find、query、select的方法都需要配置为read-only="true",否则读的时候还是从master库读数据


<tx:advice id="txAdvice" transaction-manager="transactionManager">  
        <tx:attributes>  
            <tx:method name="add*" propagation="REQUIRED" read-only="false" />  
            <tx:method name="insert*" propagation="REQUIRED" read-only="false" />  
            <tx:method name="save*" propagation="REQUIRED" read-only="false" />  
            <tx:method name="update*" propagation="REQUIRED" read-only="false" />  
            <tx:method name="modify*" propagation="REQUIRED" read-only="false" />  
            <tx:method name="edit*" propagation="REQUIRED" read-only="false" />  
            <tx:method name="delete*" propagation="REQUIRED" read-only="false" />  
            <tx:method name="remove*" propagation="REQUIRED" read-only="false" />  
            <tx:method name="active*" propagation="REQUIRED" read-only="false" />  
            <tx:method name="deactive*" propagation="REQUIRED" read-only="false" />  
            <tx:method name="set*" propagation="REQUIRED" read-only="false" />  
  
            <tx:method name="get*" read-only="true" />  
            <tx:method name="find*" read-only="true" />  
            <tx:method name="load*" read-only="true" />  
            <tx:method name="search*" read-only="true" />  
            <tx:method name="select*" read-only="true" />  
            <tx:method name="datagrid*" read-only="true" />  
            <tx:method name="query*" read-only="true" />  
            <tx:method name="*" propagation="REQUIRED" read-only="false" />  
        </tx:attributes>  
    </tx:advice>
    <!-- 定义AOP配置 -->
    <aop:config>
        <!-- 定义一个切入点 -->
        <aop:pointcut expression="execution (* com.zhou.sgl.service.*.*(..))" id="services"/>
        <!-- 对切入点和事务的通知,进行适配 -->
        <aop:advisor advice-ref="txAdvice" pointcut-ref="services"/>
    </aop:config>

好了,这样基本就实现了读写分离了,不过有个弊端,数据库在做主从复制时,多少会出现延迟,如果插入数据后立马在从库里面读数据,刚刚插入的数据可能会查不到,这就需要在业务上稍作处理了,不要插入数据后里面就查数据,主从复制是在毫秒级的,所以插入数据后1、2后再插数据完全是没问题的。

参考:http://blog.csdn.net/juan0728juan/article/details/52229720
http://blog.csdn.net/lixiucheng005/article/details/17391857 







                
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值