PHP функция для генерации V4 UUID
поэтому я немного покопался, и я пытался собрать воедино функцию, которая генерирует действительный UUID v4 в PHP. Это самое близкое, что я смог сделать. Мои знания в шестнадцатеричных, десятичных, двоичных, побитовых операторах PHP и т. п. почти не существуют. Эта функция генерирует допустимый UUID v4 вплоть до одной области. UUID v4 должен быть в виде:
xxxxxxxx-xxxx -4 xxx -yxxx-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;
}
?>
спасибо всем, кто может мне помочь.
10 ответов:
принято от этой комментарий к руководству 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 ) ); }
вместо того, чтобы разбивать его на отдельные поля, проще сгенерировать случайный блок данных и изменить отдельные позиции байтов. Вы также должны использовать лучший генератор случайных чисел, чем mt_rand ().
по данным RFC 4122-раздел 4.4, вам нужно изменить эти поля:
time_hi_and_version
(биты 4-7 7-й октет),clock_seq_hi_and_reserved
(бит 6 и 7 9-го октета)все остальные 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()
:echo guidv4(random_bytes(16));
кто с помощью композитор зависимости, вы можете рассмотреть эту библиотеку:https://github.com/ramsey/uuid
это не становится легче, чем это:
Uuid::uuid4();
в системах unix используйте ядро системы для создания uuid для вас.
file_get_contents('/proc/sys/kernel/random/uuid')
кредит Самвин на https://serverfault.com/a/529319/210994
внимание!: Использование этого метода для получения uuid фактически исчерпывает пул энтропии, очень быстро! Я бы избегал использовать это там, где он будет называться часто.
в моем поиске для создания 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)); }
фото: Павел.Волынцев
Edit: чтобы уточнить, эта функция всегда будет давать вам uuid v4 (PHP >= 5.3.0).
когда функция com_create_guid доступна (обычно только в Windows), она будет использовать это и удалять фигурные скобки.
если нет (Linux), он будет вернитесь к этой сильной случайной функции openssl_random_pseudo_bytes, затем она будет использовать vsprintf для форматирования ее в uuid v4.
мой ответ основан на комментарии комментарий пользователя 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
вдохновленный broofa ' s ответ здесь.
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');
Если вы используете
CakePHP
вы можете использовать их методCakeText::uuid();
С CakeText класс для создания RFC4122 uuid.
от Тома, на 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])