Maison >développement back-end >tutoriel php >PHP生成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的说明,怎么感觉这些生成方法 撞击率 达不到标准啊?
网上搜索,发现大体都类似如下:
<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>