1、先配置好mysql proxy方式的主从复制,参考我之前转载的文章:CentOS 7 下MySQL 5.7.12主从复制架构配置记录(亲自验证可行);
2、用mysql Connector/J 提供的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
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>
好了,这样基本就实现了读写分离了,不过有个弊端,数据库在做主从复制时,多少会出现延迟,如果插入数据后立马在从库里面读数据,刚刚插入的数据可能会查不到,这就需要在业务上稍作处理了,不要插入数据后里面就查数据,主从复制是在毫秒级的,所以插入数据后1、2后再插数据完全是没问题的。
当然这种方式也会问题,找了很多博客有比这更好的方法,但感觉稍微比较麻烦点,大家可以参考以下几篇文章看看
http://shift-alt-ctrl.iteye.com/blog/2271730
http://jinnianshilongnian.iteye.com/blog/1720618?page=2#comments
http://ahuaxuan.iteye.com/blog/205926