您现在的位置:计算机技术学习网 > 技术中心 > WEB编程 > PHP >

ThinkPHP处理海量数据分表机制详细代码及说明

来源:网络 责任编辑:栏目编辑 发表时间:2013-07-01 11:20 点击:

   应用ThinkPHP内置的分表算法处理百万级用户数据.

  数据表:

  house_member_0

  house_member_1

  house_member_2

  house_member_3

  模型中

  class MemberModel extends AdvModel {

  protected $partition = array(field=>username,type=>id,num=>4);

  public function getDao($data=array()) {

  $data = empty($data) ? $_POST : $data;

  $table = $this->getPartitionTableName($data);

  return $this->table($table);

  }

  }

  方法中

  class MemberAction extends BaseAction {

  public function login() {

  if($this->isPost()) {

  $this->validToken();

  $dao = D(Member)->getDao();

  $res = $dao->where(username = .$_POST[username])->find();

  // output 为自定义方法

  // $isAjax - bool

  $this->output(false);

  }

  $this->display();

  }

  }

  /**

  +----------------------------------------------------------

  * 得到分表的的数据表名

  +----------------------------------------------------------

  * @access public

  +----------------------------------------------------------

  * @param array $data 操作的数据

  +----------------------------------------------------------

  * @return string

  +----------------------------------------------------------

  */

  public function getPartitionTableName($data=array()) {

  // 对数据表进行分区

  if(isset($data[$this->partition[field]])) {

  $field = $data[$this->partition[field]];

  switch($this->partition[type]) {

  case id:

  // 按照id范围分表

  $step = $this->partition[expr];

  $seq = floor($field / $step)+1;

  break;

  case year:

  // 按照年份分表

  if(!is_numeric($field)) {

  $field = strtotime($field);

  }

  $seq = date(Y,$field)-$this->partition[expr]+1;

  break;

  case mod:

  // 按照id的模数分表

  $seq = ($field % $this->partition[num])+1;

  break;

  case md5:

  // 按照md5的序列分表

  $seq = (ord(substr(md5($field),0,1)) % $this->partition[num])+1;

  break;

  default :

  if(function_exists($this->partition[type])) {

  // 支持指定函数哈希

  $fun = $this->partition[type];

  $seq = (ord(substr($fun($field),0,1)) % $this->partition[num])+1;

  }else{

  // 按照字段的首字母的值分表

  $seq = (ord($field{0}) % $this->partition[num])+1;

  }

  }

  return $this->getTableName()._.$seq;

  }else{

  // 当设置的分表字段不在查询条件或者数据中

  // 进行联合查询,必须设定 partition[num]

  $tableName = array();

  for($i=0;$i<$this->partition[num];$i++)

  $tableName[] = SELECT * FROM .$this->getTableName()._.$i;

  $tableName = ( .implode(" UNION ",$tableName).) AS .$this->name;

  return $tableName;

  }

  }

    相关新闻>>

      发表评论
      请自觉遵守互联网相关的政策法规,严禁发布色情、暴力、反动的言论。
      用户名: 验证码:点击我更换图片
      最新评论 更多>>

      推荐热点

      • PHP测试
      • 十天学会php之第六天
      • 几种显示数据的方法的比较
      • 使用xmlhttp为网站增加域名查询功能
      • PHP+MYSQL+Javascript数据库查询结果的动态显示
      • 查找数组中指定键名的值
      • 用redis实现跨服务器session
      • 用新浪微博接口发送图片微博失败的原因
      • smarty局部缓存技术[源码分析]
      网站首页 - 友情链接 - 网站地图 - TAG标签 - RSS订阅 - 内容搜索
      Copyright © 2008-2015 计算机技术学习交流网. 版权所有

      豫ICP备11007008号-1