PHP如何生成唯一的数字ID,注意是数字的,不要字符串的
https://github.com/sschiau/Particle.php
PHP implementation of Twitter Snowflake ID Generator
For high availability within and across data centers, machines generating ids should not have to coordinate with each other.
PHP (tested on version 5.5.3)
id (64 bits) is composed of:
time - 41 bits (millisecond precision w/ a custom epoch)
configured machine id - 10 bits - up to 512 machines
sequence number - 12 bits - up to 2048 random numbers
You should use NTP to keep your system clock accurate.
Change const EPOCH in particle class to today epoch time w/ miliseconds (13 digits)
$machineID = 1; // Machine ID (aka Server ID no)
Particle::generateParticle($machineID);
$particleID = '4611692470816737853';
Particle::timeFromParticle($particleID);
uniqid is really good, why don't you use it ? just need convert to number , first answer is cool.
sorry I can't type Chinese, because it's disabled on chrome , I think it's a bug of segmentfault.com
我来回答,你们帮我看下我的思路,
php先生成3组数据str1{1-99},str2{100-199},str3{200-299}100递增,
程序运行取str1的第一个数字,并删除他,
当str1取完了,
去取str2的第一个数字,并删除,同时生成str1{300-399};
当str2取完了
去取str3的第一个数字,并删除,同时生成str2{400-499};
当str3用完了
去取str1的第一个数字,并删除,同时生成str3{500-599};
可以生成一个保证同一时空内不重复的uuid或者guid,这个开源的类有很多,自己也可以写,根据当前时间戳+随机因子.随机因子可包括当前的pc的mac地址和随机字符串和当前代码的行数等等,PHP内置的函数:
com_create_guid可以完成你的功能.如果你不满足,github上有许多开源的实现.
补充:1应该是要固定长度 2遇到字母转成数字或将随机引起用数字表示
贴上我一段伪代码,我使用过的id生成器
$id = substr(strtotime(date("Y-m-d", time())), 0, 2) . substr(strrev(microtime()), 0, 2) . substr(mt_rand(), 0, 5) . substr(rand(), 0, 2);
$sql = "SELECT id FROM table WHERE id=" . $id . " LIMIT 1";
$data mysql_query($sql);
思路大概是时间戳加上随机数拼凑,具体实现当然楼主根据需求再做修改
虽然根据计算基本可以得出唯一,但是不能确保,最好入库加唯一键,通过查询来保证唯一值
1 回答4k 阅读✓ 已解决
3 回答1.8k 阅读✓ 已解决
2 回答2.2k 阅读✓ 已解决
1 回答1.4k 阅读✓ 已解决
2 回答2.2k 阅读
1 回答577 阅读✓ 已解决
786 阅读
楼主,你这个问题大了。
twitter,weibo等都是专门做了一个发号器来解决这个问题的。
twitter那一套东西,叫做snowflake,楼上已经有人指出过了。这玩意一共64bit,前41bit是以微妙的时间戳,10bit是机器护着说服务器id,最后12bit是seq序列累加计数器。
weibo的方法和twitter是很类似的,将id分割为n个段,每段采集一定的数据源,最后生成一个高度唯一的id。
flickr是用的一个叫做ticketserver的玩意,使用纯mysql来实现的。
create table ticket(
id bigint unsigned not null auto_increment primary key,
stag char(1) not null default '',
unique key
stag
(stag
))engine=myisam;
先插入一条记录,然后再用replace去获取这个id。
replace into ticket(stag) values('a');
select last_insert_id();====>ID
然后是,UUID也是一个不错的选择,但是UUID生成的序列太长,而且mysql本身不具备原生支持(我假装楼主用的mysql),但楼主可以尝试把uuid当作binary(16)来保存效果会好些儿。但楼主说是纯数字,那就白搭了。
还有一种,是我群里听到的。就是提前生成一大坨可用的唯一的id,用的时候直接取然后del掉,这也是可行的,因为说这个方案的那个人是个大神。为什么提前生成呢,我觉得最主要一条就是预防高并发情况下,两个人得到的id是一样 ; 提前生成的话可以排队一个一个生成,确保唯一。当然,他们的生成策略是什么我就不太清楚了。楼主可以参考twitter的做法。
另有,mongodb自带的objectId也是一种高度唯一的序列,楼主可以利用Mongodb生成的直接拿过来用,但也要确保高并发情况下,两个人或者更多人得到同一个id,虽然概率很低。但楼主说是纯数字,那还是白搭。
如果楼主玩的单机,那就不用太纠结唯一id的问题,主键自增就可以做唯一id。太长远的问题,现在可以考虑,但不能过于拘泥。如果楼主是分布式,那就是要必须了要这个东西了。