Rumah  >  Artikel  >  pembangunan bahagian belakang  >  PHP生成UUID的正确姿势是?

PHP生成UUID的正确姿势是?

WBOY
WBOYasal
2016-12-01 00:56:292110semak imbas

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

<code>        $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;</code>
<code>            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;</code>

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

回复内容:

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

<code>        $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;</code>
<code>            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;</code>

看了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.

自行生成一个唯一值:

<code>php best.php
<?php echo uniqid(mt_rand().'_', true)."\n";
echo uniqid(mt_rand().'_', true)."\n";
//代码前后两次调用输出都不一样:
405797689_58295933a5de69.42485678
748054035_58295933a5e072.26553654
//如果运行时再加上远程用户地址/端口/进程PID的话,唯一性应该都已经够用了.
echo uniqid($_SERVER['REMOTE_ADDR'].'_'.$_SERVER['REMOTE_PORT'].'_'.getmypid().'_'.mt_rand().'_', true)."\n";
echo uniqid($_SERVER['REMOTE_ADDR'].'_'.$_SERVER['REMOTE_PORT'].'_'.getmypid().'_'.mt_rand().'_', true)."\n";</code></code>

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

<code>echo base64_encode(openssl_random_pseudo_bytes(32));</code>
Kenyataan:
Kandungan artikel ini disumbangkan secara sukarela oleh netizen, dan hak cipta adalah milik pengarang asal. Laman web ini tidak memikul tanggungjawab undang-undang yang sepadan. Jika anda menemui sebarang kandungan yang disyaki plagiarisme atau pelanggaran, sila hubungi admin@php.cn