随着一 个网站的业务不断扩展,数据不断增加,数据库的压力也会越来越大,对数据库或者SQL的基本优化可能达不到最终的效果,我们可以采用读写分离的策略来改变 现状。读写分离现在被大量应用于很多大型网站,这个技术也不足为奇了。ebay就做得非常好。ebay用的是oracle,听说是用Quest Share Plex 来实现主从复制数据。
读写分离简单的说是把对数据库读和写的操作分开对应不同的数据库服务器,这样能有效地减轻数据库压力,也能减轻io压力。主数据库提供写操作,从数据库提 供读操作,其实在很多系统中,主要是读的操作。当主数据库进行写操作时,数据要同步到从的数据库,这样才能有效保证数据库完整性。 Quest SharePlex 就是比较牛的同步数据工具,听说比oracle本身的流复制还好,MySQL也有自己的同步数据技术。mysql只要是通过二进制日志来复制数据。通过日志在从数据库重复主数据库的操作达到复制数据目的。这个复制比较好的就是通过异步方法,把数据同步到从数据库。
主数据库同步到从数据库后,从数据库一般由多台数据库组成这样才能达到减轻压力的目的。读的操作怎么样分配到从数据库上?应该根据服务器的压力把读的操作分配到服务器,而不是简单的随机分配。mysql提供了 MySQL-Proxy实现读写分离操作。不过MySQL-Proxy好像很久不更新了。oracle可以通过F5有效分配读从数据库的压力。

ebay的读写分离(网上找到就拿来用了)

mysql的读写分离
上面说的数据库同步复制,都是在从同一种数据库中,如果我要把oracle的数据同步到mysql中,其实要实现这种方案的理由很简单,mysql免费,oracle太贵。好像 Quest SharePlex 也实现不了改功能吧。好像现在市面还没有这个工具吧。那样应该怎么实现数据同步?其实我们可以考虑自己开发一套同步数据组件,通过消息,实现异步复制数据。其实这个实现起来要考虑很多方面问题,高并发的问题,失败记录等。其实这种方法也可以同步数据到 memcache 中。 听说oracle的Stream也能实现,不过没有试过。
- public function isReadOperation($sql) {
- return preg_match('/^\s*(SELECT|SHOW|DESC|PRAGMA)\s+/i',$sql);
- }
php读写分离类
- <?php
-
-
-
-
-
-
-
-
- class mysql_rw_php {
-
-
- var $querynum = 0;
-
- var $link = null;
-
- var $charset;
-
- var $cur_db = '';
-
-
- var $ro_exist = false;
-
- var $link_ro = null;
-
- var $link_rw = null;
-
- function mysql_rw_php(){
- }
-
- function connect($dbhost, $dbuser, $dbpw, $dbname = '', $pconnect = 0, $halt = TRUE) {
- if($pconnect) {
- if(!$this->link = @mysql_pconnect($dbhost, $dbuser, $dbpw)) {
- $halt && $this->halt('Can not connect to MySQL server');
- }
- } else {
- if(!$this->link = @mysql_connect($dbhost, $dbuser, $dbpw)) {
- $halt && $this->halt('Can not connect to MySQL server');
- }
- }
-
-
- if(!$this->link && !$halt) return false;
-
-
- if($this->link_rw == null)
- $this->link_rw = $this->link;
-
- if($this->version() > '4.1') {
- if($this->charset) {
- @mysql_query("SET character_set_connection=$this->charset, character_set_results=$this->charset, character_set_client=binary", $this->link);
- }
- if($this->version() > '5.0.1') {
- @mysql_query("SET sql_mode=''", $this->link);
- }
- }
- if($dbname) {
- $this->select_db($dbname);
- }
- }
-
-
- function connect_ro($dbhost, $dbuser, $dbpw, $dbname = '', $pconnect = 0){
- if($this->link_rw == null)
- $this->link_rw = $this->link;
- $this->link = null;
-
- $this->connect($dbhost, $dbuser, $dbpw, $dbname, $pconnect, false);
- if($this->link){
-
-
- $this->ro_exist = true;
- $this->link_ro = $this->link;
- if($this->cur_db){
-
- @mysql_select_db($this->cur_db, $this->link_ro);
- }
- }else{
-
-
- $this->link = &$this->link_rw;
- }
- }
-
-
- function set_ro_list($ro_list){
- if(is_array($ro_list)){
-
- $link_ro = $ro_list[array_rand($ro_list)];
- $this->connect_ro($link_ro['dbhost'], $link_ro['dbuser'], $link_ro['dbpw']);
- }
- }
-
- function select_db($dbname) {
-
- $this->cur_db = $dbname;
- if($this->ro_exist){
- @mysql_select_db($dbname, $this->link_ro);
- }
- return @mysql_select_db($dbname, $this->link_rw);
- }
-
- function fetch_array($query, $result_type = MYSQL_ASSOC) {
- return mysql_fetch_array($query, $result_type);
- }
-
- function fetch_one_array($sql, $type = '') {
- $qr = $this->query($sql, $type);
- return $this->fetch_array($qr);
- }
-
- function query($sql, $type = '') {
- $this->link = &$this->link_rw;
-
- if($this->ro_exist && preg_match ("/^(\s*)select/i", $sql)){
- $this->link = &$this->link_ro;
- }
- $func = $type == 'UNBUFFERED' && @function_exists('mysql_unbuffered_query') ?
- 'mysql_unbuffered_query' : 'mysql_query';
- if(!($query = $func($sql, $this->link)) && $type != 'SILENT') {
- $this->halt('MySQL Query Error', $sql);
- }
- $this->querynum++;
- return $query;
- }
-
- function affected_rows() {
- return mysql_affected_rows($this->link);
- }
-
- function error() {
- return (($this->link) ? mysql_error($this->link) : mysql_error());
- }
-
- function errno() {
- return intval(($this->link) ? mysql_errno($this->link) : mysql_errno());
- }
-
- function result($query, $row) {
- $query = @mysql_result($query, $row);
- return $query;
- }
-
- function num_rows($query) {
- $query = mysql_num_rows($query);
- return $query;
- }
-
- function num_fields($query) {
- return mysql_num_fields($query);
- }
-
- function free_result($query) {
- return mysql_free_result($query);
- }
-
- function insert_id() {
- return ($id = mysql_insert_id($this->link)) >= 0 ? $id : $this->result($this->query("SELECT last_insert_id()"), 0);
- }
-
- function fetch_row($query) {
- $query = mysql_fetch_row($query);
- return $query;
- }
-
- function fetch_fields($query) {
- return mysql_fetch_field($query);
- }
-
- function version() {
- return mysql_get_server_info($this->link);
- }
-
- function close() {
- return mysql_close($this->link);
- }
-
- function halt($message = '', $sql = '') {
- $dberror = $this->error();
- $dberrno = $this->errno();
- echo "<div style=\"position:absolute;font-size:11px;font-family:verdana,arial;background:#EBEBEB;padding:0.5em;\">
- <b>MySQL Error</b><br>
- <b>Message</b>: $message<br>
- <b>SQL</b>: $sql<br>
- <b>Error</b>: $dberror<br>
- <b>Errno.</b>: $dberrno<br>
- </div>";
- exit();
- }
- }
-
- ?>
调用方法
- <?php
-
-
-
-
-
-
-
-
- require_once('mysql_rw_php.class.php');
-
-
- $db_rw = array(
- 'dbhost'=>'www.aslibra.com',
- 'dbuser'=>'aslibra',
- 'dbpw'=>'www.aslibra.com',
- 'dbname'=>'test'
- );
-
- $db_ro = array(
- array(
- 'dbhost'=>'www.aslibra.com:4306',
- 'dbuser'=>'aslibra',
- 'dbpw'=>'www.aslibra.com'
- )
- );
-
- $DB = new mysql_rw_php;
-
-
- $DB->connect($db_rw[dbhost], $db_rw[dbuser], $db_rw[dbpw], $db_rw[dbname]);
-
-
- $DB->connect_ro($db_ro[0][dbhost], $db_ro[0][dbuser], $db_ro[0][dbpw]);
-
-
- $DB->set_ro_list($db_ro);
-
-
- $sql = "insert into a set a='test'";
- $DB->query($sql);
-
-
- $sql = "select * from a";
- $qr = $DB->query($sql);
- while($row = $DB->fetch_array($qr)){
- echo $row[a];
- }
- ?>