创建以人为本,非连续的唯一id的良好实践

Eve*_*ert 9 php mysql

我需要为用户创建一个唯一的ID.我不想使用auto_incrementing id,因为我不希望用户能够猜出我们拥有多少用户,或者增长率是多少.

UUID也不是一个选项,因为用户必须在智能手机上重新输入ID.

所以我希望我能尽可能地减少"强制"数据库以找到未使用的ID.什么是一个聪明的方法来解决这个问题?

谢谢!

Kai*_*aii 19

好的,我会再试一次.你的目标是:

  • 模糊标识符的增量率 - 不是连续的,不可猜测的,不可计算的
  • 对于智能手机用户来说仍然"可用",因此长UUID或SHA1哈希超出了范围
  • 避免对数据库进行随机猜测/强力计算的必要性
  • 您的外键仍具有良好的数据库性能

要在数据库中保持良好的性能,您应该为PK保留标准的auto_incrementing整数.请注意,InnoDB根据PK对行进行排序(请参阅" InnoDB聚集索引 "),因此如果使用某种魔术哈希作为PK,则最终会在聚簇索引中进行大量重新排序,从而导致写入性能下降.

我建议使用加密算法(加密,而不是散列)来加密和解密您想要混淆的ID.此外,为了获得最佳混淆,您希望对结果字符串使用最小长度.生成的字符串应该仍然可用,因此您必须使用类似base64的内容来呈现用户可读的内容.

试试这个加密示例:

function my_number_encrypt($data, $key, $base64_safe=true, $minlen=8) {
        $data = base_convert($data, 10, 36);
        $data = str_pad($data, $minlen, '0', STR_PAD_LEFT);
        $data = @mcrypt_encrypt(MCRYPT_BLOWFISH, $key, $data, MCRYPT_MODE_CBC);
        if ($base64_safe) $data = str_replace('=', '', base64_encode($data));
        return $data;
}

function my_number_decrypt($data, $key, $base64_safe=true) {
        if ($base64_safe) $data = base64_decode($data.'==');
        $data = @mcrypt_decrypt(MCRYPT_BLOWFISH, $key, $data, MCRYPT_MODE_CBC);
        $data = base_convert($data, 36, 10);
        return $data;
}

$key = "my super secret magic bytes";

$id = 12345678; // obtain_normal_key_from_mysql();

// give ID to user
$enc = my_number_encrypt($i, $key);

// get ID from user
$dec = my_number_decrypt($enc, $key);
// fetch from database using normal ID -> 12345678

// demo code
for($i=10000; $i<10050; $i++) {
        $enc = my_number_encrypt($i, $key);
        $dec = my_number_decrypt($enc, $key);
        echo "$i -> $enc -> $dec", PHP_EOL;
}
Run Code Online (Sandbox Code Playgroud)

演示结果:

10000 -> 1RXK468NYes -> 10000
10001 -> QdEov5mjMPA -> 10001
10002 -> 2gsgzWJgD+8 -> 10002
10003 -> 2zwPwhqr9HI -> 10003
10004 -> Xq+kDh1UFuM -> 10004
10005 -> wfwv6TrW9xY -> 10005
10006 -> 1Lck1L0HJ/U -> 10006
10007 -> v+3YY2zfL1A -> 10007
10008 -> 5AmGlqD8byM -> 10008
10009 -> pZBIpPnKXHU -> 10009
10010 -> CAeWdKGkk8c -> 10010
10011 -> fYddnLOSK6U -> 10011
10012 -> na8Ry0erHv8 -> 10012
10013 -> zxNj+ZJVMBY -> 10013
10014 -> gWJWC9VulZc -> 10014
10015 -> 5pR9B79eM/E -> 10015
10016 -> MQtpBhpzHRA -> 10016
10017 -> dW+3nejBEIg -> 10017
10018 -> znB/feM6104 -> 10018
10019 -> RtdRwwRyEcs -> 10019
10020 -> 4cW/OWT140E -> 10020
10021 -> dIvK9VjOevg -> 10021
10022 -> QxLdfrucc/Y -> 10022
10023 -> M0KN3sX10Gs -> 10023
10024 -> 827yFJyDCG4 -> 10024
10025 -> JF/VRj92qL8 -> 10025
10026 -> IXTvn/SCzek -> 10026
10027 -> L4nFwvhgwX8 -> 10027
10028 -> z0lve9nhgDA -> 10028
10029 -> m/UBgZzfIXo -> 10029
10030 -> IfWcrLKTHXk -> 10030
10031 -> n/jPFwKR/9A -> 10031
10032 -> j1mm2kbeWl0 -> 10032
10033 -> cm7mOQMVa6k -> 10033
10034 -> jCUuweEyRME -> 10034
10035 -> LDaMcOWKxjg -> 10035
10036 -> Zcrd5XzhhIk -> 10036
10037 -> j0Yg/fCjyAA -> 10037
10038 -> /LmlvRHmmmg -> 10038
10039 -> t0juuzGSKs4 -> 10039
10040 -> 9CoRCVXaak4 -> 10040
10041 -> tFmImR4j0JM -> 10041
10042 -> nI3Thy51hLg -> 10042
10043 -> mTCJh0/h2mE -> 10043
10044 -> S196xdyb3Os -> 10044
10045 -> ItOyUp+J4Q4 -> 10045
10046 -> DL87SidiOLM -> 10046
10047 -> d+Nw3xBqV44 -> 10047
10048 -> 3YzVelaC4uI -> 10048
10049 -> fAUJVOl6PaU -> 10049
Run Code Online (Sandbox Code Playgroud)


Mch*_*chl 5

创建一个随机uniqe数字列表(您可以使用PHP range()shuffle()函数),并将其存储在数据库中,甚至存储在txt文件中.确保列表足够长,以便持续一段时间.然后,只要您需要新ID,只需弹出列表中的第一个值即可.

$list = range(0,999999); //list now contains numbers from 0 to 999999 
// if you ever need to add more ID's to your list, start at 1000000.
shuffle($list); //now it's randomly ordered
Run Code Online (Sandbox Code Playgroud)

  • 对此的扩展是使用S/KEY字典映射生成的ID以使其非常容易键入(32位意味着3个短字)faqs.org/rfcs/rfc1760.html具有字典. (2认同)