最近一个项目因为数据量变大,须要进行数据分表。数据存储在淘宝的tddl上。分表后,原先的自增id就不能使用了。tddl对java支持很好,分表后无需考虑全局id的问题。可是这个项目使用的是php进行开发,必须本身生成全局id。如下列出几种分表方案,仅当抛砖引玉。php
其实这里并非严格的CAS,而是使用了比较交换原子操做的思想。
生成思路以下:每次生成全局id时,先从sequence表中获取当前的全局最大id。而后在获取的全局id上作加1操做。把加1后的值更新到数据库。更新时是关键。
如加1后的值为203,表名是users,数据表结构以下:java
CREATE TABLE `SEQUENCE` ( `name` varchar(30) NOT NULL COMMENT '分表的表名', `gid` bigint(20) NOT NULL COMMENT '最大全局id', PRIMARY KEY (`name`) ) E
那么更新语句是。
update sequence set gid = 203 where name = 'users' and gid < 203;
sql语句的 and gid < 203 是为了保证并发环境下gid的值只增不减。
若是update语句的影响记录条数为0说明,已经有其余进程提早生成了203这个值,并写入了数据库。须要重复以上步骤重新生成。
代码实现以下:redis
//$name 表名 function next_id_db($name){ //获取数据库全局sequence对象 $seq_dao = Wk_Sequence_Dao_Sequence::getInstance(); $threshold = 100; //最大尝试次数 for($i = 0; $i < $threshold; $i++){ $last_id = $seq_dao->get_seq_id($name);//从数据库获取全局id $id = $last_id +1; $ret = $seq_dao->set_seq_id($name, $id); if($ret){ return $id; break; } } return false; }
在进行并发编程时,通常都会使用锁机制。其实,全局id的生成也是解决并发问题。
生成思路以下:
在使用redis的setnx方法和memcace的add方法时,若是指定的key已经存在,则返回false。利用这个特性,实现全局锁。
每次生成全局id前,先检测指定的key是否存在。
若是不存在则使用redis的incr方法或者memcache的increment进行加1操做。这两个方法的返回值是加1后的值。
若是存在,则程序进入循环等待状态。循环过程当中不断检测key是否还存在,若是key不存在就执行上面的操做。
代码以下:sql
//使用redis实现 //$name 为 逻辑表名 function next_id_redis($name){ $redis = Wk_Redis_Util::getRedis();//获取redis对象 $seq_dao = Wk_Sequence_Dao_Sequence::getInstance();//获取存储全局id数据表对象 if(!is_object($redis)){ throw new Exception("fail to create redis object"); } $max_times = 10; //最大执行次数 避免redis不可用的时候 进入死循环 while(1){ $i++; //检测key是否存在,至关于检测锁是否存在 $ret = $redis->setnx("sequence_{$name}_flag",time()); if($ret){ break; } if($i > $max_times){ break; } $time = $redis->get("sequence_{$name}_flag"); if(is_numeric($time) && time() - $time > 1){//若是循环等待时间大于1秒,则再也不等待。 break; } } $id = $redis->incr("sequence_{$name}"); //若是操做失败,则从sequence表中获取全局id并加载到redis if (intval($id) === 1 or $id === false) { $last_id = $seq_dao->get_seq_id($name);//从数据库获取全局id if(!is_numeric($last_id)){ throw new Exception("fail to get id from db"); } $ret = $redis->set("sequence_{$name}",$last_id); if($ret == false){ throw new Exception("fail to set redis key [ sequence_{$name} ]"); } $id = $redis->incr("sequence_{$name}"); if(!is_numeric($id)){ throw new Exception("fail to incr redis key [ sequence_{$name} ]"); } } $seq_dao->set_seq_id($name, $id);//把生成的全局id写入数据表sequence $redis->delete("sequence_{$name}_flag");//删除key,至关于释放锁 $db = null; return $id; }
使用redis直接操做内存,可能性能会好些。可是若是redis死掉后,如何处理呢?把以上两种方案结合,提供更好的稳定性。
代码以下:数据库
function next_id($name){ try{ return $this->next_id_redis($name); } catch(Exception $e){ return $this->next_id_db($name); } }
另外对于全局id的生成,Flicker和Twitter也都公布了本身的方案。感兴趣的人,能够了解下。编程