欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  后端开发

PHP生成UUID的正确姿势是?

程序员文章站 2022-03-25 11:57:21
...
网上搜索,发现大体都类似如下:
        $chars = md5(uniqid(mt_rand(), true));
        $uuid  = substr($chars,0,8) . '-';
        $uuid .= substr($chars,8,4) . '-';
        $uuid .= substr($chars,12,4) . '-';
        $uuid .= substr($chars,16,4) . '-';
        $uuid .= substr($chars,20,12);
        return $prefix . $uuid;
            mt_srand ( ( double ) microtime () * 10000 ); //optional for php 4.2.0 and up.随便数播种,4.2.0以后不需要了。
            $charid = strtoupper ( md5 ( uniqid ( rand (), true ) ) ); //根据当前时间(微秒计)生成唯一id.
            $hyphen = chr ( 45 );
            $uuid = '' . 
                substr ( $charid, 0, 8 ) . $hyphen . substr ( $charid, 8, 4 ) . $hyphen . substr ( $charid, 12, 4 ) . $hyphen . substr ( $charid, 16, 4 ) . $hyphen . substr ( $charid, 20, 12 );
            return $uuid;

看了UUID的说明,怎么感觉这些生成方法 撞击率 达不到标准啊?

回复内容:

网上搜索,发现大体都类似如下:

        $chars = md5(uniqid(mt_rand(), true));
        $uuid  = substr($chars,0,8) . '-';
        $uuid .= substr($chars,8,4) . '-';
        $uuid .= substr($chars,12,4) . '-';
        $uuid .= substr($chars,16,4) . '-';
        $uuid .= substr($chars,20,12);
        return $prefix . $uuid;
            mt_srand ( ( double ) microtime () * 10000 ); //optional for php 4.2.0 and up.随便数播种,4.2.0以后不需要了。
            $charid = strtoupper ( md5 ( uniqid ( rand (), true ) ) ); //根据当前时间(微秒计)生成唯一id.
            $hyphen = chr ( 45 );
            $uuid = '' . 
                substr ( $charid, 0, 8 ) . $hyphen . substr ( $charid, 8, 4 ) . $hyphen . substr ( $charid, 12, 4 ) . $hyphen . substr ( $charid, 16, 4 ) . $hyphen . substr ( $charid, 20, 12 );
            return $uuid;

看了UUID的说明,怎么感觉这些生成方法 撞击率 达不到标准啊?

目前很多所谓PHP生成UUID的方法都以随机数最为其算法的基础,这虽然在实践中很难出现重复的几率,但在理论上显然是行不通的。

这里用MongoDb的ObjectId作为例子,个人觉得这种生成方法是可以引入到PHP中的。
ObjectId主要分为四块:时间戳、毫秒数、机器码、自增计数。

前两者就不用细说了,PHP都可以取到,重点在后两者上。

机器码主要用于避免在不同主机间生成相同的UUID。
机器码目前主要的生成方法是根据硬件信息进行散列计算出,而这个方法并不适用于PHP。
主要因为两点,一是PHP不能直接获得硬件信息(需要通过扩展或者执行本地命令等方法),二是PHP请求与请求之间无法共享数据,所以每次请求都得拉取机器码,对性能影响很大。
解决这一问题有个很简单的办法,就是在不同机器先计算出机器码,之后直接写入配置文件,PHP处理时直接读取即可。

自增计数主要用于避免在并发处理中生成相同的UUID。
对于自增来说,PHP由于不能在请求间共享数据,所以不能直接实现自增计数。
解决这一问题我们可以引入PHP的两个扩展,分别是Semaphore和Shared Memory,这两者都在PHP源码中集成,通过--enable-sysvsem和--enable-sysvshm就能打开。我们只需要把自增数放置在共享内存中,再通过信号量限制访问,即可达到并发请求共享自增技术的目的了。当然,自增也能通过其他工具来实现。

Linux内核提供有UUID生成接口:
cat /proc/sys/kernel/random/uuid
Linux上一切皆文件,不管什么程序,读取文件就能获取一个UUID.

自行生成一个唯一值:

php best.php

另外貌似用openssl_random_pseudo_bytes也能生成不错的随机串:

echo base64_encode(openssl_random_pseudo_bytes(32));
相关标签: uuid php