springmvc+mybatis+druid+mysql proxy读写分离配置

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


其中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>

好了,这样基本就实现了读写分离了,不过有个弊端,数据库在做主从复制时,多少会出现延迟,如果插入数据后立马在从库里面读数据,刚刚插入的数据可能会查不到,这就需要在业务上稍作处理了,不要插入数据后里面就查数据,主从复制是在毫秒级的,所以插入数据后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

相关文章
相关标签/搜索