用户表分表原理

当用户数量达到百万甚至千万级别的时候,可能没法靠单标知足现有需求。此时经常使用的处理方式就是分库分表。这里介绍一个简单的分表方法。sql

对用户进行分表,首先要解决的是将一个用户分配到那个表,而后如何知道去哪一个表查询该用户的数据。解决了这个两个问题。不管将该用户的数据放在哪一个表都无所谓。数据库

首先来看一个方法:ui

function getHash($uid) {

    return sprintf('%03x',intval(sprintf('%u', crc32($uid)))%512 );

}

 

该方法完成了如下步骤:spa

(1)经过循环冗余校验,每一个字符串都会生成一个固定的数字。code

(2)用该数字对512取模获得一个0<=X<512的数字server

(3)再用16进制格式化,可生成一个固定的3个字符的字符串。blog

(4)该方法的做用能够经过用户ID生成一个固定的3个字符的字符串。根据该字符串可将该用户分配到对应的表中。ci

 

不论用户id是什么,经过循环冗余校验后都能获得一个数字。将该数字取模。而后将余数格式化成字符串。(固然也能够不格式化,直接用数字也能够。)而后用该字符串作前缀或者后缀建立用户表。字符串

 

例如:get

我先以固定字符串为后缀,在user数据库建立了user_000、user_001 … user_1ff共512张用户表。(固然512张表的结构是如出一辙的。)

$model = new Model();
$db = 'userdb';
for
($i=0; $i<512; $i++){ $num = sprintf('%03x', $i); $sql="CREATE TABLE IF NOT EXISTS `{$db}`.`user_{$num}` ( `uid` bigint(20) NOT NULL, `name` int(11) NOT NULL, `server` int(11) NOT NULL, `cid` int(11) NOT NULL, `aid` int(11) NOT NULL, `areaId` int(11) NOT NULL, `eventtime` int(11) NOT NULL, `eventdate` date NOT NULL, `eventhour` int(11) NOT NULL, PRIMARY KEY (`uid`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8"; $model->query($sql); }

 

而后将一个用户ID为test@163.com 经过以上行数进行格式化,获得一个3个字符串:1c8 ,而后将此用户的数据存入user.user_1c8 数据表中,之后须要用到这个用户的数据,只要进该用户id也就是test@163.com 经过以上方法获得1c8 而后从数据表user.user_1c8 中查询该用户的数据便可。

相关文章
相关标签/搜索