278

所以我一直在做一些挖掘工作,我一直在尝试拼凑一个在 PHP 中生成有效 v4 UUID 的函数。这是我能来的最近的一次。我在十六进制、十进制、二进制、PHP 的位运算符等方面的知识几乎不存在。此函数生成一个有效的 v4 UUID,直到一个区域。v4 UUID 应采用以下形式:

xxxxxxxx-xxxx- 4 xxx- y xxx-xxxxxxxxxxxx

其中y 是 8、9、A 或 B。这是函数失败的地方,因为它不遵守这一点。

我希望在这方面比我有更多知识的人可以帮助我并帮助我修复此功能,使其符合该规则。

功能如下:

<?php

function gen_uuid() {
 $uuid = array(
  'time_low'  => 0,
  'time_mid'  => 0,
  'time_hi'  => 0,
  'clock_seq_hi' => 0,
  'clock_seq_low' => 0,
  'node'   => array()
 );
 
 $uuid['time_low'] = mt_rand(0, 0xffff) + (mt_rand(0, 0xffff) << 16);
 $uuid['time_mid'] = mt_rand(0, 0xffff);
 $uuid['time_hi'] = (4 << 12) | (mt_rand(0, 0x1000));
 $uuid['clock_seq_hi'] = (1 << 7) | (mt_rand(0, 128));
 $uuid['clock_seq_low'] = mt_rand(0, 255);
 
 for ($i = 0; $i < 6; $i++) {
  $uuid['node'][$i] = mt_rand(0, 255);
 }
 
 $uuid = sprintf('%08x-%04x-%04x-%02x%02x-%02x%02x%02x%02x%02x%02x',
  $uuid['time_low'],
  $uuid['time_mid'],
  $uuid['time_hi'],
  $uuid['clock_seq_hi'],
  $uuid['clock_seq_low'],
  $uuid['node'][0],
  $uuid['node'][1],
  $uuid['node'][2],
  $uuid['node'][3],
  $uuid['node'][4],
  $uuid['node'][5]
 );
 
 return $uuid;
}

?>
4

17 回答 17

419

与其将其分解为单个字段,不如生成随机数据块并更改单个字节位置更容易。您还应该使用比 mt_rand() 更好的随机数生成器。

根据RFC 4122 - Section 4.4,您需要更改这些字段:

  1. time_hi_and_version(第 7 个八位字节的第 4-7 位),
  2. clock_seq_hi_and_reserved(第 9 个八位字节的第 6 位和第 7 位)

所有其他 122 位都应该足够随机。

以下方法使用 生成 128 位随机数据openssl_random_pseudo_bytes()​​,对八位字节进行排列,然后使用bin2hex()vsprintf()进行最终格式化。

function guidv4($data)
{
    assert(strlen($data) == 16);

    $data[6] = chr(ord($data[6]) & 0x0f | 0x40); // set version to 0100
    $data[8] = chr(ord($data[8]) & 0x3f | 0x80); // set bits 6-7 to 10

    return vsprintf('%s%s-%s-%s-%s-%s%s%s', str_split(bin2hex($data), 4));
}

echo guidv4(openssl_random_pseudo_bytes(16));

使用 PHP 7,生成随机字节序列更加简单random_bytes()

function guidv4($data = null)
{
    $data = $data ?? random_bytes(16);
    // ...
}
于 2013-04-08T09:21:10.553 回答
322

取自PHP 手册的评论,您可以使用:

function gen_uuid() {
    return sprintf( '%04x%04x-%04x-%04x-%04x-%04x%04x%04x',
        // 32 bits for "time_low"
        mt_rand( 0, 0xffff ), mt_rand( 0, 0xffff ),

        // 16 bits for "time_mid"
        mt_rand( 0, 0xffff ),

        // 16 bits for "time_hi_and_version",
        // four most significant bits holds version number 4
        mt_rand( 0, 0x0fff ) | 0x4000,

        // 16 bits, 8 bits for "clk_seq_hi_res",
        // 8 bits for "clk_seq_low",
        // two most significant bits holds zero and one for variant DCE1.1
        mt_rand( 0, 0x3fff ) | 0x8000,

        // 48 bits for "node"
        mt_rand( 0, 0xffff ), mt_rand( 0, 0xffff ), mt_rand( 0, 0xffff )
    );
}
于 2010-01-11T06:27:39.670 回答
139

任何使用作曲家依赖项的人,您可能需要考虑这个库:https ://github.com/ramsey/uuid

没有比这更容易的了:

Uuid::uuid4();
于 2014-07-16T16:23:25.770 回答
32

在 unix 系统上,使用系统内核为您生成一个 uuid。

file_get_contents('/proc/sys/kernel/random/uuid')

在https://serverfault.com/a/529319/210994上信用 Samveen

注意!:使用这种方法获取 uuid 实际上会很快耗尽熵池!我会避免在经常调用它的地方使用它。

于 2014-04-17T19:04:55.020 回答
16

Jack 的回答略有不同,以添加对 PHP < 7 的支持:

// Get an RFC-4122 compliant globaly unique identifier
function get_guid() {
    $data = PHP_MAJOR_VERSION < 7 ? openssl_random_pseudo_bytes(16) : random_bytes(16);
    $data[6] = chr(ord($data[6]) & 0x0f | 0x40);    // Set version to 0100
    $data[8] = chr(ord($data[8]) & 0x3f | 0x80);    // Set bits 6-7 to 10
    return vsprintf('%s%s-%s-%s-%s-%s%s%s', str_split(bin2hex($data), 4));
}
于 2019-03-31T09:48:01.443 回答
16
// php version >= 7
$uuid = vsprintf('%s%s-%s-%s-%s-%s%s%s', str_split(bin2hex(random_bytes(16)), 4));
于 2020-01-16T18:49:23.727 回答
13

在我搜索创建 v4 uuid 时,我首先来到了这个页面,然后在http://php.net/manual/en/function.com-create-guid.php上找到了这个

function guidv4()
{
    if (function_exists('com_create_guid') === true)
        return trim(com_create_guid(), '{}');

    $data = openssl_random_pseudo_bytes(16);
    $data[6] = chr(ord($data[6]) & 0x0f | 0x40); // set version to 0100
    $data[8] = chr(ord($data[8]) & 0x3f | 0x80); // set bits 6-7 to 10
    return vsprintf('%s%s-%s-%s-%s-%s%s%s', str_split(bin2hex($data), 4));
}

信用:pavel.volyntsev

编辑:澄清一下,这个函数总是会给你一个 v4 uuid (PHP >= 5.3.0)。

当 com_create_guid 函数可用时(通常仅在 Windows 上),它将使用该函数并去除花括号。

如果不存在(Linux),它将依赖这个强大的随机 openssl_random_pseudo_bytes 函数,然后它将使用 vsprintf 将其格式化为 v4 uuid。

于 2017-06-12T17:03:44.497 回答
5

我的回答是基于评论uniqid 用户评论,但它使用openssl_random_pseudo_bytes函数生成随机字符串而不是从/dev/urandom

function guid()
{
    $randomString = openssl_random_pseudo_bytes(16);
    $time_low = bin2hex(substr($randomString, 0, 4));
    $time_mid = bin2hex(substr($randomString, 4, 2));
    $time_hi_and_version = bin2hex(substr($randomString, 6, 2));
    $clock_seq_hi_and_reserved = bin2hex(substr($randomString, 8, 2));
    $node = bin2hex(substr($randomString, 10, 6));

    /**
     * Set the four most significant bits (bits 12 through 15) of the
     * time_hi_and_version field to the 4-bit version number from
     * Section 4.1.3.
     * @see http://tools.ietf.org/html/rfc4122#section-4.1.3
    */
    $time_hi_and_version = hexdec($time_hi_and_version);
    $time_hi_and_version = $time_hi_and_version >> 4;
    $time_hi_and_version = $time_hi_and_version | 0x4000;

    /**
     * Set the two most significant bits (bits 6 and 7) of the
     * clock_seq_hi_and_reserved to zero and one, respectively.
     */
    $clock_seq_hi_and_reserved = hexdec($clock_seq_hi_and_reserved);
    $clock_seq_hi_and_reserved = $clock_seq_hi_and_reserved >> 2;
    $clock_seq_hi_and_reserved = $clock_seq_hi_and_reserved | 0x8000;

    return sprintf('%08s-%04s-%04x-%04x-%012s', $time_low, $time_mid, $time_hi_and_version, $clock_seq_hi_and_reserved, $node);
} // guid
于 2013-04-08T08:27:52.167 回答
5

如果您使用CakePHP,您可以使用 CakeText 类中的方法CakeText::uuid();生成RFC4122 uuid。

于 2017-09-07T18:33:20.410 回答
4

受到broofa的回答的启发

preg_replace_callback('/[xy]/', function ($matches)
{
  return dechex('x' == $matches[0] ? mt_rand(0, 15) : (mt_rand(0, 15) & 0x3 | 0x8));
}
, 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx');

或者如果无法使用匿名函数。

preg_replace_callback('/[xy]/', create_function(
  '$matches',
  'return dechex("x" == $matches[0] ? mt_rand(0, 15) : (mt_rand(0, 15) & 0x3 | 0x8));'
)
, 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx');
于 2013-03-22T12:34:52.790 回答
4

在搜索了完全相同的东西并几乎自己实现了一个版本之后,我认为值得一提的是,如果您在WordPress框架内执行此操作,WP 有自己的超级方便的功能:

$myUUID = wp_generate_uuid4();

您可以在此处阅读说明和来源。

于 2018-10-12T19:36:07.743 回答
4

使用Symfony Polyfill / Uuid
然后你可以生成 uuid 作为原生 php 函数:

$uuid = uuid_create(UUID_TYPE_RANDOM);

有关它的更多信息,请阅读官方 Symfony blop 帖子 - https://symfony.com/blog/introducing-the-new-symfony-uuid-polyfill

于 2020-05-14T09:58:00.220 回答
2

使用 mysql 为您生成 uuid 怎么样?

$conn = new mysqli($servername, $username, $password, $dbname, $port);

$query = 'SELECT UUID()';
echo $conn->query($query)->fetch_row()[0];
于 2017-05-26T01:47:46.450 回答
1

来自汤姆,在http://www.php.net/manual/en/function.uniqid.php

$r = unpack('v*', fread(fopen('/dev/random', 'r'),16));
$uuid = sprintf('%04x%04x-%04x-%04x-%04x-%04x%04x%04x',
    $r[1], $r[2], $r[3], $r[4] & 0x0fff | 0x4000,
    $r[5] & 0x3fff | 0x8000, $r[6], $r[7], $r[8])
于 2012-08-22T18:12:50.793 回答
1

我确信有一种更优雅的方法可以将4xxxyxxx部分从二进制转换为十进制。但是,如果您想openssl_random_pseudo_bytes用作密码安全的数字生成器,这就是我使用的:

return sprintf('%s-%s-%04x-%04x-%s',
    bin2hex(openssl_random_pseudo_bytes(4)),
    bin2hex(openssl_random_pseudo_bytes(2)),
    hexdec(bin2hex(openssl_random_pseudo_bytes(2))) & 0x0fff | 0x4000,
    hexdec(bin2hex(openssl_random_pseudo_bytes(2))) & 0x3fff | 0x8000,
    bin2hex(openssl_random_pseudo_bytes(6))
    );
于 2019-01-25T21:14:56.890 回答
0

这可以更简单吗?

$uuid = bin2hex(openssl_random_pseudo_bytes(16));
for($cnt = 8; $cnt <=23; $cnt+=5)
   $uuid = substr($uuid, 0, $cnt) . "-" . substr($uuid, $cnt);

echo $uuid . "\n";
于 2021-09-11T02:44:23.003 回答
0

只是一个想法,但我最终获得 V4 GUID 的方法是使用数据库服务器。我正在使用 SQL Server,在需要 GUID 的场景中,我已经在运行查询,所以我只是将 newid() 添加为查询结果字段之一。这给了我需要的 V4 GUID。

这显然取决于您使用的数据库服务器,以及您需要 GUID 的代码中的其他内容(以及您需要多少 GUID),但是如果您的数据库服务器生成 v4 GUID,尤其是如果您正在运行无论如何查询,这是获取 GUID 的一种快速且简单的独立于 PHP 版本的方法。

于 2021-09-20T16:29:12.837 回答