質問
それで、私はいくつかの掘り下げをしてきましたが、PHPで有効なV4 UUIDを生成する関数をつなぎ合わせようとしています。これは私が来ることができた最も近いものです。ヘックス、小数、バイナリ、PHPのビットワイズ演算子などでの私の知識は、ほとんど存在しません。この関数は、1つの領域まで有効なV4 UUIDを生成します。 v4 uuidは次の形である必要があります。
xxxxxxxx-xxxx-4xxx-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;
}
?>
私を助けることができる人に感謝します。
解決
から取られた これ 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
(第7オクテットの4-7)、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);
// ...
}
誰でも使用しています 作曲 依存関係、このライブラリを検討することをお勧めします。 https://github.com/ramsey/uuid
これよりも簡単になりません:
Uuid::uuid4();
UNIXシステムでは、システムカーネルを使用してUUIDを生成します。
file_get_contents('/proc/sys/kernel/random/uuid')
Samveenをクレジットします 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));
}
クレジット:pavel.volyntsev
編集:明確にするために、この関数は常にV4 UUID(php> = 5.3.0)を提供します。
com_create_guid関数が利用可能になると(通常はWindowsでのみ)、それを使用して巻き毛のブレースを剥がします。
存在しない場合(Linux)、この強力なランダムOpenSSL_RANDOM_PSEUDO_BYTES関数に頼ります。その後、VSPRINTFを使用してV4 UUIDにフォーマットします。
私の答えはコメントに基づいています 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
使用する場合 CakePHP
それらの方法を使用できます CakeText::uuid();
から caketext RFC4122 UUIDを生成するクラス。
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');
まったく同じことを検索した ほとんど 自分でこれのバージョンを実装すると、あなたがこれをしているなら、それは言及する価値があると思いました WordPress フレームワーク、WPには、まさにこれのための独自の非常に便利な機能があります。
$myUUID = wp_generate_uuid4();
説明とソースを読むことができます ここ.
わずかなバリエーション ジャックの答え 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));
}
トムから、オン 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])
MySQLを使用してUUIDを生成してみませんか?
$conn = new mysqli($servername, $username, $password, $dbname, $port);
$query = 'SELECT UUID()';
echo $conn->query($query)->fetch_row()[0];
バイナリからデシマルへの変換を行うには、よりエレガントな方法があると確信しています 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))
);