您好,欢迎访问一九零五行业门户网

mysql读写分离实战-构建高性能web的代码示例

一个完整的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的代码示例的详细内容。
其它类似信息

推荐信息