请教 PHP 有什么的好的办法生成流水号?

2015-09-03 00:01:23 +08:00
 tanteng

比如流水号是 10 位: 1000000001 , 1000000002 ,这样的,我想使用 PHP 生成这样的流水号,根据我的理解,流水号应该是连续增长的,但是我每次生成的时候怎么不重复而且保持连续呢?

存数据库的话比较麻烦,而且每次要查库,还要更新数据,性能低。也不能使用随机码,这样数字不连续。放到 redis 里面万一 redis 挂掉了呢所以也不好。

PHP 生成流水号有什么好办法?

10509 次点击
所在节点    PHP
24 条回复
guoer
2015-09-03 00:08:31 +08:00
redis 可以备份的
dsgygb
2015-09-03 00:13:00 +08:00
窝这有个生成订单的函数。=--是随机的。只能保证 26 年内不重复。。。
function make_number (){
$year_code = array ('A','B','C','D','E','F','G','H','I','J','K','L','M','N','O','P','Q','I','S','T','U','V','W','X','Y','Z');



return $year_code[intval (date ('Y'))-2015].
strtoupper (dechex (date ('m'))).date ('d').
substr (time (),-5 ).substr (microtime (),2,5 ).sprintf ('%02d',rand (0,99 ));


}
oott123
2015-09-03 00:15:10 +08:00
不依赖数据库、不依赖外部存储…
万一楼主的 php 挂了呢?
realpg
2015-09-03 00:22:09 +08:00
不依赖数据库不依赖外部存储?
既然你叫流水号要求不冲突,

如果是顺序的,那么至少要有一个地方存储当前值或者下一个值吧?
如果是随机的,那么至少要有一个地方存储已经生成的全部号码避免生成重复吧?
branchzero
2015-09-03 00:26:01 +08:00
满足这个条件的思路:
得有一个进程锁,保证每次读取 ID 之前,前面跑脚本的操作已经完全进行完毕了(也就是在每次读取 ID 前上锁, ID 自增后解锁)。
然后随便找一个靠谱地方存计数器就是了(文件啊、数据库、内存缓存或者其他,随便你)

用数据库的话上面这些都不是问题,不用数据库的话,自己想办法解决吧。
br00k
2015-09-03 00:29:00 +08:00
万一服务器挂了呢。
incompatible
2015-09-03 00:29:58 +08:00
每次生成一批放在缓存里 随用随取
快用完时再生成一批
wumch
2015-09-03 00:32:09 +08:00
你这是同时高要求一致性、可用性、分区容错性啊
lavadore
2015-09-03 00:35:31 +08:00
数据库已经有这个功能了,你不用为什么非要在 php 里用?

数据库不用查库, auto_increment 每次插入后自动返回流水号,已有的功能为什么不用
Mac
2015-09-03 00:45:05 +08:00
@dsgygb 直接用年份岂不是可以保证永久不重复?高并发的情况下也不能保证吧,只不过有这么大的订单量环境很罕见而已
aprikyblue
2015-09-03 00:47:33 +08:00
万一来一道雷呢
cevincheung
2015-09-03 00:55:28 +08:00
mysql
select uuid_short ()

生成流水号也是用来入库的,数据库挂了生成流水号也没啥用。
huigeer
2015-09-03 01:55:09 +08:00
时间戳加随机数
rails
2015-09-03 03:34:29 +08:00
其实楼主的问题可以简化为:如何在 php 内计数
1004
2015-09-03 03:35:34 +08:00
弄个订单系统不就行了
roychan
2015-09-03 07:58:15 +08:00
要不查库那肯定只能找一个自递增或者自递减的变量了,也就是时间,自己设计一个算法生成不会重复的流水号即可。
moe3000
2015-09-03 08:14:23 +08:00
redis 中开一个 key ,每次取出后, incr 一次,直接用 reids 生成
不过这样就依赖 redis 了
tanteng
2015-09-03 09:09:07 +08:00
@moe3000 redis 重启或挂了会不会丢失之前保存的值
wwek
2015-09-03 09:38:28 +08:00
@tanteng redis 支撑持久化啊
tanteng
2015-09-03 16:04:24 +08:00
@moe3000 那我想通过 redis 自增的方式获取流水号,线上 redis 有没有可能重启或断点,那流水号就重新开始了

这是一个专为移动设备优化的页面(即为了让你能够在 Google 搜索结果里秒开这个页面),如果你希望参与 V2EX 社区的讨论,你可以继续到 V2EX 上打开本讨论主题的完整版本。

https://www.v2ex.com/t/217976

V2EX 是创意工作者们的社区,是一个分享自己正在做的有趣事物、交流想法,可以遇见新朋友甚至新机会的地方。

V2EX is a community of developers, designers and creative people.

© 2021 V2EX