Hashids 是一个用于生成短、唯一的非连续ID的库,它可以将整数(如数据库中的自增ID)转换为唯一的字符串。在 PHP 中,你可以使用 hashids/hashids
这个包来实现这个功能。
首先,安装 Hashids:
composer require hashids/hashids
接下来,我们来看一下 Hashids 的基本用法:
encode($id); // 转换为字符串 echo $hash . PHP_EOL; // 输出字符串 $decodedId = $hashids->decode($hash)[0]; // 解码回整数 echo $decodedId . PHP_EOL; // 输出整数
现在,我们来分析 Hashids 的源码。首先,创建一个新的 Hashids 实例时,会传入一些参数,如下所示:
public function __construct(
string $salt = '',
int $minHashLength = 0,
string $alphabet = 'abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ1234567890'
) {
// ...
}
$salt
:用于加密的盐值,可以为空。$minHashLength
:生成的哈希字符串的最小长度。$alphabet
:用于生成哈希字符串的字符集。
接下来,我们来看一下 encode()
和 decode()
方法的实现。这里只列出关键部分:
public function encode(...$numbers): string
{
// ...
while (count($numbers)) {
$number = array_shift($numbers);
$buffer = '';
do {
$buffer .= $this->alphabet[$number % $this->alphabetLength];
$number = ($number - ($number % $this->alphabetLength)) / $this->alphabetLength;
} while ($number > 0);
$result = strrev($buffer) . $result;
}
// ...
return $result;
}
public function decode(string $hash): array
{
// ...
$hashArray = str_split(strrev($hash));
foreach ($hashArray as $i => $char) {
$number += strpos($this->alphabet, $char) * pow($this->alphabetLength, $i);
}
// ...
return $result;
}
encode()
方法将整数转换为字符串,它首先计算余数并将其添加到缓冲区,然后将结果反转并与之前的结果拼接。decode()
方法则是将字符串反转后,计算每个字符在字母表中的位置,然后将其相加得到原始整数。
修改建议:
- 更改默认的字母表:可以在创建 Hashids 实例时传入自定义的字母表,以增加哈希字符串的复杂性。
$hashids = new Hashids('', 0, 'abcdefghijklmnopqrstuvwxyz!@#$%^&*()');
- 使用盐值:为了提高安全性,可以在创建 Hashids 实例时传入一个盐值。这样,即使两个人使用相同的字母表,他们也会得到不同的哈希字符串。
$hashids = new Hashids('my_salt');
- 设置最小哈希长度:可以通过设置最小哈希长度来增加哈希字符串的长度,从而提高安全性。
$hashids = new Hashids('', 10);
总之,Hashids 是一个很好的库,可以帮助你生成短、唯一的非连续ID。你可以根据需要对其进行修改和优化。