PHP функция для генерации v4 UUID - PullRequest
203 голосов
/ 11 января 2010

Итак, я немного покопался и пытался собрать воедино функцию, которая генерирует действительный UUID v4 в PHP. Это самое близкое, что мне удалось приехать. Мои знания в шестнадцатеричном, десятичном, двоичном, побитовых операторах PHP и тому подобном практически не существуют. Эта функция генерирует действительный UUID v4 вплоть до одной области. UUID версии 4 должен иметь вид:

XXXXXXXX-xxxx- 4 xxx- у ххх-ХХХХХХХХХХХХ

, где 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;
}

?>

Спасибо всем, кто может мне помочь.

Ответы [ 13 ]

1 голос
/ 26 мая 2017

Как насчет использования mysql для генерации uuid для вас?

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

$query = 'SELECT UUID()';
echo $conn->query($query)->fetch_row()[0];
1 голос
/ 22 августа 2012

С Тома, на 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])
0 голосов
/ 26 января 2019

Я уверен, что есть более элегантный способ преобразования двоичного числа в десятичное для порций 4xxx и yxxx. Но если вы хотите использовать 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))
    );
...