thinkphp下MySQL数据库读写分离代码剖析


Posted in PHP onApril 18, 2017

当采用原生态的sql语句进行写入操作的时候,要用execute,读操作要用query。

MySQL数据主从同步还是要靠MySQL的机制来实现,所以这个时候MySQL主从同步的延迟问题是需要优化,延迟时间太长不仅影响业务,还影响用户体验。

thinkphp核心类Thinkphp/library/Model.class.php 中,query 方法,调用Thinkphp/library/Think/Db/Driver/Mysql.class.php

/**
   * SQL查询
   * @access public
   * @param string $sql SQL
   * @param mixed $parse 是否需要解析SQL 
   * @return mixed
   */
  public function query($sql,$parse=false) {
    if(!is_bool($parse) && !is_array($parse)) {
      $parse = func_get_args();
      array_shift($parse);
    }
    $sql =  $this->parseSql($sql,$parse);
    return $this->db->query($sql);
  }

调用Thinkphp/library/Think/Db/Driver/Mysql.class.php

/**
   * 执行查询 返回数据集
   * @access public
   * @param string $str sql指令
   * @return mixed
   */
  public function query($str) {
    if(0===stripos($str, 'call')){ // 存储过程查询支持
      $this->close();
      $this->connected  =  false;
    }
    $this->initConnect(false);
    if ( !$this->_linkID ) return false;
    $this->queryStr = $str;
    //释放前次的查询结果
    if ( $this->queryID ) {  $this->free();  }
    N('db_query',1);
    // 记录开始执行时间
    G('queryStartTime');
    $this->queryID = mysql_query($str, $this->_linkID);
    $this->debug();
    if ( false === $this->queryID ) {
      $this->error();
      return false;
    } else {
      $this->numRows = mysql_num_rows($this->queryID);
      return $this->getAll();
    }
  }

上面初始化数据库链接时,initConnect(false),调用Thinkphp/library/Think/Db/Db.class.php,注意false、true代码实现。true表示直接调用主库,false表示调用读写分离的读库。

/**
   * 初始化数据库连接
   * @access protected
   * @param boolean $master 主服务器
   * @return void
   */
  protected function initConnect($master=true) {
    if(1 == C('DB_DEPLOY_TYPE'))
      // 采用分布式数据库
      $this->_linkID = $this->multiConnect($master);
    else
      // 默认单数据库
      if ( !$this->connected ) $this->_linkID = $this->connect();
  }

  /**
   * 连接分布式服务器
   * @access protected
   * @param boolean $master 主服务器
   * @return void
   */
  protected function multiConnect($master=false) {
    foreach ($this->config as $key=>$val){
      $_config[$key]   =  explode(',',$val);
    }    
    // 数据库读写是否分离
    if(C('DB_RW_SEPARATE')){
      // 主从式采用读写分离
      if($master)
        // 主服务器写入
        $r =  floor(mt_rand(0,C('DB_MASTER_NUM')-1));
      else{
        if(is_numeric(C('DB_SLAVE_NO'))) {// 指定服务器读
          $r = C('DB_SLAVE_NO');
        }else{
          // 读操作连接从服务器
          $r = floor(mt_rand(C('DB_MASTER_NUM'),count($_config['hostname'])-1));  // 每次随机连接的数据库
        }
      }
    }else{
      // 读写操作不区分服务器
      $r = floor(mt_rand(0,count($_config['hostname'])-1));  // 每次随机连接的数据库
    }
    $db_config = array(
      'username' => isset($_config['username'][$r])?$_config['username'][$r]:$_config['username'][0],
      'password' => isset($_config['password'][$r])?$_config['password'][$r]:$_config['password'][0],
      'hostname' => isset($_config['hostname'][$r])?$_config['hostname'][$r]:$_config['hostname'][0],
      'hostport' => isset($_config['hostport'][$r])?$_config['hostport'][$r]:$_config['hostport'][0],
      'database' => isset($_config['database'][$r])?$_config['database'][$r]:$_config['database'][0],
      'dsn'    => isset($_config['dsn'][$r])?$_config['dsn'][$r]:$_config['dsn'][0],
      'params'  => isset($_config['params'][$r])?$_config['params'][$r]:$_config['params'][0],
      'charset'  => isset($_config['charset'][$r])?$_config['charset'][$r]:$_config['charset'][0],      
    );
    return $this->connect($db_config,$r);
  }

query方法参数为false,其他删除、更新、增加读主库。这一点可以结合Thinkphp/library/Model.class.php中的delete、save、add操作,参数为true。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

PHP 相关文章推荐
用Flash图形化数据(二)
Oct 09 PHP
php mssql 日期出现中文字符的解决方法
Mar 10 PHP
使用php+Ajax实现唯一校验实现代码[简单应用]
Nov 29 PHP
PHP得到mssql的存储过程的输出参数功能实现
Nov 23 PHP
shell脚本作为保证PHP脚本不挂掉的守护进程实例分享
Jul 15 PHP
php模拟登陆的实现方法分析
Jan 09 PHP
PHP SPL标准库中的常用函数介绍
May 11 PHP
CodeIgniter钩子用法实例详解
Jan 20 PHP
PHP 在数组中搜索给定的简单实例 array_search 函数
Jun 13 PHP
Laravel+Intervention实现上传图片功能示例
Jul 09 PHP
yii框架数据库关联查询操作示例
Oct 14 PHP
PHP笛卡尔积实现原理及代码实例
Dec 09 PHP
Thinkphp通过一个入口文件如何区分移动端和PC端
Apr 18 #PHP
Yii2汉字转拼音类的实例代码
Apr 18 #PHP
php+resumablejs实现的分块上传 断点续传功能示例
Apr 18 #PHP
ZendFramework2连接数据库操作实例
Apr 18 #PHP
PHP实现的数独求解问题示例
Apr 18 #PHP
PHP使用finfo_file()函数检测上传图片类型的实现方法
Apr 18 #PHP
php实现不通过扩展名准确判断文件类型的方法【finfo_file方法与二进制流】
Apr 18 #PHP
You might like
PHP实现采集程序原理和简单示例代码
2007/03/18 PHP
php 表单验证实现代码
2009/03/10 PHP
探讨PHP删除文件夹的三种方法
2013/06/09 PHP
php生成随机密码自定义函数代码(简单快速)
2014/05/10 PHP
php pthreads多线程的安装与使用
2016/01/19 PHP
简单实用的PHP文本缓存类实例
2019/03/22 PHP
javascript firefox不显示本地预览图片问题的解决方法
2008/11/12 Javascript
JQuery 学习笔记 选择器之一
2009/07/23 Javascript
Javascript 面向对象 继承
2010/05/13 Javascript
一个简单的瀑布流效果(主体形式自写)
2013/05/27 Javascript
node.js使用npm 安装插件时提示install Error: ENOENT报错的解决方法
2014/11/20 Javascript
node.js中的querystring.stringify方法使用说明
2014/12/10 Javascript
详解参数传递四种形式
2015/07/21 Javascript
JS实现超简单的仿QQ折叠菜单效果
2015/09/21 Javascript
jQuery插件JWPlayer视频播放器用法实例分析
2017/01/11 Javascript
jQuery插件HighCharts实现的2D面积图效果示例【附demo源码下载】
2017/03/15 Javascript
微信小程序实现瀑布流布局与无限加载的方法详解
2017/05/12 Javascript
微信小程序适配iphoneX的实现方法
2018/09/18 Javascript
JQuery Ajax执行跨域请求数据的解决方案
2018/12/10 jQuery
微信小程序实现二维码签到考勤系统
2020/01/16 Javascript
JavaScript运动原理基础知识详解
2020/04/02 Javascript
Angular进行简单单元测试的实现方法实例
2020/08/16 Javascript
python切换hosts文件代码示例
2013/12/31 Python
跟老齐学Python之折腾一下目录
2014/10/24 Python
python线程池(threadpool)模块使用笔记详解
2017/11/17 Python
在Windows中设置Python环境变量的实例讲解
2018/04/28 Python
python中类的属性和方法介绍
2018/11/27 Python
django 自定义过滤器(filter)处理较为复杂的变量方法
2019/08/12 Python
2014年党务公开实施方案
2014/02/27 职场文书
2014财务年度工作总结
2014/11/11 职场文书
2015年消费者权益日活动总结
2015/02/09 职场文书
给校长的建议书范文
2015/09/14 职场文书
2016年社区国庆节活动总结
2016/04/01 职场文书
民政局2016年“六一”儿童节慰问活动总结
2016/04/06 职场文书
tensorflow+k-means聚类简单实现猫狗图像分类的方法
2021/04/28 Python
Python通过loop.run_in_executor执行同步代码 同步变为异步
2022/04/11 Python