ruby base64 通过以 2 位字符开头来编码 128 位数字,以防止末尾填充

2024-04-04

这个问题是我之前问题的后续:如何在 Ruby 中使用自定义字符集将 UUID 转换为字符串? https://stackoverflow.com/questions/47803212/how-can-i-convert-a-uuid-to-a-string-using-a-custom-character-set-in-ruby但我会尝试将其表述为一个单独且具体的问题。

我确实有一个 Ruby 128 位 UUID 作为十六进制值:

SecureRandom.uuid #=> "2d931510-d99f-494a-8c67-87feb05e1594"

如果我正确地获得了 IFC 规范(http://www.buildingsmart-tech.org/ifc/IFC2x3/TC1/html/ifcutilityresource/lexical/ifcgloballyuniqueid.htm http://www.buildingsmart-tech.org/ifc/IFC2x3/TC1/html/ifcutilityresource/lexical/ifcgloballyuniqueid.htm),我想对此进行 Base64 编码,但我希望输出以 2 位字符(4 个选项)开始,而不是 6 位(64 个选项所需)开始,而不是在末尾进行填充。

这样我想我最终可以得到一个 22 个字符的字符串(1 个 2 位,21 个 6 位,总共 128 位)。

是否可以通过这种方式调整 Ruby base64?


简短的回答:不。从技术上讲,这不是标准的 Base64,因此 Ruby 的标准库不会处理它。

Ruby 的 Base64 lib 将其输入视为字节,因此您需要让输入数据能被 8 整除。但是您需要在 UUID 前面有 4 个零位,所以这是 4+128=132,所以下一个最接近的 8 倍数是136 即 17 字节。您可以在最后丢弃额外的随机性:

x = SecureRandom.gen_random 17   # get a little extra randomness
x[0] = (x[0].ord & 0x0f).chr     # 0 out the first four bits
Base64.strict_encode64(x)[0...22] # discard extra randomness

这种方法的一个缺点是你的 128 位 UUID 在内部奇怪地对齐x并且很难单独看到。如果你想取出 128 位,你可以通过一些打包/解包来实现:

[x.unpack("B*")[0][4...132]].pack("B*")
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

ruby base64 通过以 2 位字符开头来编码 128 位数字,以防止末尾填充 的相关文章

随机推荐