一个完整的mysql读写分离环境包括以下几个部分:
应用程序client
database proxy
database集群
在本次实战中,应用程序client基于c3p0连接后端的database proxy。database proxy负责管理client实际访问database的路由策略,采用开源框架amoeba。database集群采用mysql的master-slave的replication方案。整个环境的结构图如下所示:
实战步骤与详解
一.搭建mysql的master-slave环境
1)分别在host1(10.20.147.110)和host2(10.20.147.111)上安装mysql(5.0.45),具体安装方法可见官方文档
2)配置master
首先编辑/etc/my.cnf,添加以下配置:
log-bin=mysql-bin #slave会基于此log-bin来做replication
server-id=1 #master的标示
binlog-do-db = amoeba_study #用于master-slave的具体数据库
然后添加专门用于replication的用户:
mysql> grant replication slave on *.* to repl@10.20.147.111 identified by '111111';
重启mysql,使得配置生效:
/etc/init.d/mysqld restart
最后查看master状态:
3)配置slave
首先编辑/etc/my.cnf,添加以下配置:
server-id=2 #slave的标示
配置生效后,配置与master的连接:
mysql> change master to
-> master_host='10.20.147.110',
-> master_user='repl',
-> master_password='111111',
-> master_log_file='mysql-bin.000003',
->
master_log_pos=161261;
其中master_host是master机的ip,master_user和master_password就是我们刚才在master上添加的用户,master_log_file和master_log_pos对应与master status里的信息
最后启动slave:
mysql> start slave;
4)验证master-slave搭建生效
通过查看slave机的log(/var/log/mysqld.log):
100703 10:51:42 [note] slave i/o thread: connected to master 'repl@10.20.147.110:3306',
replication started in log 'mysql-bin.000003' at position 161261
如看到以上信息则证明搭建成功,如果有问题也可通过此log找原因
二.搭建database proxy
此次实战中database proxy采用amoeba
,它的相关信息可以查阅官方文档,不在此详述
1)安装amoeba
下载amoeba(1.2.0-ga)后解压到本地(d:/opensource/amoeba-mysql-1.2.0-ga),即完成安装
2)配置amoeba
先配置proxy连接和与各后端mysql服务器连接信息(d:/opensource/amoeba-mysql-1.2.0-ga/conf/amoeba.xml):
以上是proxy提供给client的连接配置
<dbserverlist>
<dbserver name="server1">
<!-- poolableobjectfactory实现类 -->
<factoryconfig class="com.meidusa.amoeba.mysql
<a href="http://lib.csdn.net/base/dotnet" class='replace_word' title=".net知识库" target='_blank' style='color:#df3434; font-weight:bold;'>.net</a>
.mysqlserverconnectionfactory">
<property name="manager">defaultmanager</property>
<!-- 真实mysql数据库端口 -->
<property name="port">3306</property>
<!-- 真实mysql数据库ip -->
<property name="ipaddress">10.20.147.110</property>
<property name="schema">amoeba_study</property>
<!-- 用于登陆mysql的用户名 -->
<property name="user">root</property>
<!-- 用于登陆mysql的密码 -->
<property name="password"></property>
</factoryconfig>
<!-- objectpool实现类 -->
<poolconfig class="com.meidusa.amoeba
<a href="http://lib.csdn.net/base/dotnet" class='replace_word' title=".net知识库" target='_blank' style='color:#df3434; font-weight:bold;'>.net</a>
.poolable.poolableobjectpool">
<property name="maxactive">200</property>
<property name="maxidle">200</property>
<property name="minidle">10</property>
<property name="minevictableidletimemillis">600000</property>
<property name="timebetweenevictionrunsmillis">600000</property>
<property name="testonborrow">true</property>
<property name="testwhileidle">true</property>
</poolconfig>
</dbserver>
<dbserver name="server2">
<!-- poolableobjectfactory实现类 -->
<factoryconfig class="com.meidusa.amoeba.mysql.net.mysqlserverconnectionfactory">
<property name="manager">defaultmanager</property>
<!-- 真实mysql数据库端口 -->
<property name="port">3306</property>
<!-- 真实mysql数据库ip -->
<property name="ipaddress">10.20.147.111</property>
<property name="schema">amoeba_study</property>
<!-- 用于登陆mysql的用户名 -->
<property name="user">root</property>
<!-- 用于登陆mysql的密码 -->
<property name="password"></property>
</factoryconfig>
<!-- objectpool实现类 -->
<poolconfig class="com.meidusa.amoeba.net.poolable.poolableobjectpool">
<property name="maxactive">200</property>
<property name="maxidle">200</property>
<property name="minidle">10</property>
<property name="minevictableidletimemillis">600000</property>
<property name="timebetweenevictionrunsmillis">600000</property>
<property name="testonborrow">true</property>
<property name="testwhileidle">true</property>
</poolconfig>
</dbserver>
</dbserverlist>
以上是proxy与后端各mysql数据库服务器配置信息,具体配置见注释很明白了
最后配置读写分离策略:
从以上配置不然发现,写操作路由到server1(master),读操作路由到server2(slave)
3)启动amoeba
在命令行里运行d:/opensource/amoeba-mysql-1.2.0-ga/amoeba.bat即可:
log4j:warn log4j config load completed from file:d:/opensource/amoeba-mysql-1.2.0-ga/conf/log4j.xml
log4j:warn ip access config load completed from file:d:/opensource/amoeba-mysql-1.2.0-ga/conf/access_list.conf
2010-07-03 09:55:33,821 info net.serverableconnectionmanager - server listening on 0.0.0.0/0.0.0.0:8066.
三.client端调用与测试
1)编写client调用程序
具体程序细节就不详述了,只是一个最普通的基于mysql driver的jdbc的数据库操作程序
2)配置数据库连接
本client基于c3p0,具体数据源配置如下:
<bean id="datasource" class="com.mchange.v2.c3p0.combopooleddatasource"
destroy-method="close">
<property name="driverclass" value="com.mysql.jdbc.driver" />
<property name="jdbcurl" value="jdbc:mysql://localhost:8066/amoeba_study" />
<property name="user" value="root" />
<property name="password" value="root" />
<property name="minpoolsize" value="1" />
<property name="maxpoolsize" value="1" />
<property name="maxidletime" value="1800" />
<property name="acquireincrement" value="1" />
<property name="maxstatements" value="0" />
<property name="initialpoolsize" value="1" />
<property name="idleconnectiontestperiod" value="1800" />
<property name="acquireretryattempts" value="6" />
<property name="acquireretrydelay" value="1000" />
<property name="breakafteracquirefailure" value="false" />
<property name="testconnectiononcheckout" value="true" />
<property name="testconnectiononcheckin" value="false" />
</bean>
值得注意是,client端只需连到proxy,与实际的数据库没有任何关系,因此jdbcurl、user、password配置都对应于amoeba暴露出来的配置信息
3)调用与测试
首先插入一条数据:
insert into zone_by_id(id,name) values(20003,'name_20003')
通过查看master机上的日志/var/lib/mysql/mysql_log.log:
100703 11:58:42 1 query set names latin1
1 query set names latin1
1 query set character_set_results = null
1 query show variables
1 query show collation
1 query set autocommit=1
1 query set sql_mode='strict_trans_tables'
1 query show variables like 'tx_isolation'
1 query show full tables from `amoeba_study` like 'probablynot'
1 prepare [1] insert into zone_by_id(id,name) values(?,?)
1 prepare [2] insert into zone_by_id(id,name) values(?,?)
1 execute [2] insert into zone_by_id(id,name) values(20003,'name_20003')
得知写操作发生在master机上
通过查看slave机上的日志/var/lib/mysql/mysql_log.log:
100703 11:58:42 2 query insert into zone_by_id(id,name) values(20003,'name_20003')
得知slave同步执行了这条语句
然后查一条数据:select t.name from zone_by_id t where t.id = 20003
通过查看slave机上的日志/var/lib/mysql/mysql_log.log:
100703 12:02:00 33 query set names latin1
33 prepare [1] select t.name from zone_by_id t where t.id = ?
33 prepare [2] select t.name from zone_by_id t where t.id = ?
33 execute [2] select t.name from zone_by_id t where t.id = 20003
得知读操作发生在slave机上
并且通过查看slave机上的日志/var/lib/mysql/mysql_log.log发现这条语句没在master上执行
通过以上验证得知简单的master-slave搭建和实战得以生效
以上就是mysql读写分离实战-构建高性能web的代码示例的详细内容。