そこで、いろいろ調べて、PHP で有効な v4 UUID を生成する関数をまとめようとしました。これが私ができた中で最も近いものです。16 進数、10 進数、2 進数、PHP のビット演算子などについての私の知識はほとんどありません。この関数は、1 つの領域まで有効な v4 UUID を生成します。v4 UUID は次の形式になります。
xxxxxxxxx-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;
}
?>
ベストアンサー1
UUIDv4 を構築するには、128 ビット相当のランダム データを生成し、いくつかのフィールドをパッチしてデータが標準に準拠するようにし、16 進グループとしてフォーマットします。
によるとRFC 4122 - セクション 4.4以下のビットを変更する必要があります:
time_hi_and_version
(第7オクテットのビット4-7)clock_seq_hi_and_reserved
(第9オクテットのビット6と7)
以下のコードは与えられたデータの順列を作成し、bin2hex()
そしてvsprintf()
最終的なフォーマットを行います。
function uuidv4()
{
$data = 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));
}
PHP 5以前
このrandom_bytes
関数は PHP 7.0 で導入されました。古いバージョンの PHP でこれを実装する方法の例を次に示します。
$data = openssl_random_pseudo_bytes(16, $strong);
// ensure the result is cryptographically strong
assert($data !== false && $strong);
ランダムデータの生成
暗号的に強力なランダムデータを生成するには、 を使用する代わりにmt_rand()
、次のいずれかを使用することをお勧めします。openssl_random_pseudo_bytes()
またはrandom_bytes()
(php7 以降) ランダム データを生成する別の方法を選択することもできます。