模块 Random::Formatter
随机数格式化器。¶ ↑
以多种方式格式化生成的随机数。当 'random/formatter'
被引入时,一些方法被添加到空的核心模块 Random::Formatter
中,使它们可以作为 Random 的实例和模块方法使用。
标准库 SecureRandom
也扩展了该模块,下面描述的方法可以在其中作为模块方法使用。
示例¶ ↑
生成随机十六进制字符串
require 'random/formatter' prng = Random.new prng.hex(10) #=> "52750b30ffbc7de3b362" prng.hex(10) #=> "92b15d6c8dc4beb5f559" prng.hex(13) #=> "39b290146bea6ce975c37cfc23" # or just Random.hex #=> "1aed0c631e41be7f77365415541052ee"
生成随机 base64 字符串
prng.base64(10) #=> "EcmTPZwWRAozdA==" prng.base64(10) #=> "KO1nIU+p9DKxGg==" prng.base64(12) #=> "7kJSM/MzBJI+75j8" Random.base64(4) #=> "bsQ3fQ=="
生成随机二进制字符串
prng.random_bytes(10) #=> "\016\t{\370g\310pbr\301" prng.random_bytes(10) #=> "\323U\030TO\234\357\020\a\337" Random.random_bytes(6) #=> "\xA1\xE6Lr\xC43"
生成字母数字字符串
prng.alphanumeric(10) #=> "S8baxMJnPl" prng.alphanumeric(10) #=> "aOxAg8BAJe" Random.alphanumeric #=> "TmP9OsJHJLtaZYhP"
生成 UUID
prng.uuid #=> "2d931510-d99f-494a-8c67-87feb05e1594" prng.uuid #=> "bad85eb9-0713-4da7-8d36-07a8e4b00eab" Random.uuid #=> "f14e0271-de96-45cc-8911-8910292a42cd"
所有方法都可以在标准库 SecureRandom
中使用
SecureRandom.hex #=> "05b45376a30c67238eb93b16499e50cf"
在给定范围内生成随机数,就像 Random
一样
prng.random_number #=> 0.5816771641321361 prng.random_number(1000) #=> 485 prng.random_number(1..6) #=> 3 prng.rand #=> 0.5816771641321361 prng.rand(1000) #=> 485 prng.rand(1..6) #=> 3
常量
- ALPHANUMERIC
alphanumeric
的默认字符列表。
公共实例方法
生成一个随机的字母数字字符串。
参数 n 指定要生成的字母数字字符串的长度(以字符为单位)。参数 chars 指定结果字符串包含的字符列表。
如果 n 未指定或为 nil,则假定为 16。将来它可能会更大。
结果可能包含 A-Z、a-z 和 0-9,除非指定了 chars。
require 'random/formatter' Random.alphanumeric #=> "2BuBuLf3WfSKyQbR" # or prng = Random.new prng.alphanumeric(10) #=> "i6K93NdqiH" Random.alphanumeric(4, chars: [*"0".."9"]) #=> "2952" # or prng = Random.new prng.alphanumeric(10, chars: [*"!".."/"]) #=> ",.,++%/''."
# File lib/random/formatter.rb, line 368 def alphanumeric(n = nil, chars: ALPHANUMERIC) n = 16 if n.nil? choose(chars, n) end
生成一个随机的 base64 字符串。
参数 n 指定要生成的随机数的长度(以字节为单位)。结果字符串的长度约为 n 的 4/3。
如果 n 未指定或为 nil,则假定为 16。将来它可能会更大。
结果可能包含 A-Z、a-z、0-9、“+”、“/” 和 “=”。
require 'random/formatter' Random.base64 #=> "/2BuBuLf3+WfSKyQbRcc/A==" # or prng = Random.new prng.base64 #=> "6BbW0pxO0YENxn38HMUbcQ=="
有关 base64 的定义,请参见 RFC 3548。
# File lib/random/formatter.rb, line 114 def base64(n=nil) [random_bytes(n)].pack("m0") end
生成一个随机的十六进制字符串。
参数n指定要生成的随机数的字节长度。生成的十六进制字符串的长度是n的两倍。
如果 n 未指定或为 nil,则假定为 16。将来它可能会更大。
结果可能包含 0-9 和 a-f。
require 'random/formatter' Random.hex #=> "eb693ec8252cd630102fd0d0fb7c3485" # or prng = Random.new prng.hex #=> "91dc3bfb4de5b11d029d376634589b61"
# File lib/random/formatter.rb, line 92 def hex(n=nil) random_bytes(n).unpack1("H*") end
从原始随机字节生成格式化的随机数。参见 Random#rand
.
生成一个随机的二进制字符串。
参数n指定结果字符串的长度。
如果n未指定或为 nil,则假定为 16。将来可能会更大。
结果可能包含任何字节:“x00” - “xff”。
require 'random/formatter' Random.random_bytes #=> "\xD8\\\xE0\xF4\r\xB2\xFC*WM\xFF\x83\x18\xF45\xB6" # or prng = Random.new prng.random_bytes #=> "m\xDC\xFC/\a\x00Uf\xB2\xB2P\xBD\xFF6S\x97"
# File lib/random/formatter.rb, line 71 def random_bytes(n=nil) n = n ? n.to_int : 16 gen_random(n) end
从原始随机字节生成格式化的随机数。参见 Random#rand
.
static VALUE rand_random_number(int argc, VALUE *argv, VALUE obj) { rb_random_t *rnd = try_get_rnd(obj); VALUE v = rand_random(argc, argv, obj, rnd); if (NIL_P(v)) v = rand_random(0, 0, obj, rnd); else if (!v) invalid_argument(argv[0]); return v; }
生成一个随机的 URL 安全的 base64 字符串。
参数 n 指定要生成的随机数的长度(以字节为单位)。结果字符串的长度约为 n 的 4/3。
如果 n 未指定或为 nil,则假定为 16。将来它可能会更大。
布尔参数padding指定填充。如果它为 false 或 nil,则不生成填充。否则生成填充。默认情况下,不生成填充,因为“=”可能用作 URL 分隔符。
结果可能包含 A-Z、a-z、0-9、“-” 和“_”。如果padding为 true,则也使用“=”。
require 'random/formatter' Random.urlsafe_base64 #=> "b4GOKm4pOYU_-BOXcrUGDg" # or prng = Random.new prng.urlsafe_base64 #=> "UZLdOkzop70Ddx-IJR0ABg" prng.urlsafe_base64(nil, true) #=> "i0XQ-7gglIsHGV2_BNPrdQ==" prng.urlsafe_base64(nil, true) #=> "-M8rLhr7JEpJlqFGUMmOxg=="
有关 URL 安全 base64 的定义,请参见 RFC 3548。
# File lib/random/formatter.rb, line 145 def urlsafe_base64(n=nil, padding=false) s = [random_bytes(n)].pack("m0") s.tr!("+/", "-_") s.delete!("=") unless padding s end
生成一个随机的 v4 UUID(通用唯一标识符)。
require 'random/formatter' Random.uuid #=> "2d931510-d99f-494a-8c67-87feb05e1594" Random.uuid #=> "bad85eb9-0713-4da7-8d36-07a8e4b00eab" # or prng = Random.new prng.uuid #=> "62936e70-1815-439b-bf89-8492855a7e6b"
版本 4 UUID 纯粹是随机的(版本除外)。它不包含有意义的信息,例如 MAC 地址、时间戳等。
结果包含 122 个随机位(15.25 个随机字节)。
有关 UUID 的详细信息,请参见 RFC4122。
# File lib/random/formatter.rb, line 169 def uuid ary = random_bytes(16).unpack("NnnnnN") ary[2] = (ary[2] & 0x0fff) | 0x4000 ary[3] = (ary[3] & 0x3fff) | 0x8000 "%08x-%04x-%04x-%04x-%04x%08x" % ary end
生成一个随机的 v7 UUID(通用唯一标识符)。
require 'random/formatter' Random.uuid_v7 # => "0188d4c3-1311-7f96-85c7-242a7aa58f1e" Random.uuid_v7 # => "0188d4c3-16fe-744f-86af-38fa04c62bb5" Random.uuid_v7 # => "0188d4c3-1af8-764f-b049-c204ce0afa23" Random.uuid_v7 # => "0188d4c3-1e74-7085-b14f-ef6415dc6f31" # |<--sorted-->| |<----- random ---->| # or prng = Random.new prng.uuid_v7 # => "0188ca51-5e72-7950-a11d-def7ff977c98"
版本 7 UUID 以 64 位 Unix 时间戳(自纪元以来的毫秒数)的最低有效 48 位开始,并用随机数据填充剩余的位,排除版本和变体位。
这允许版本 7 UUID 按创建时间排序。 Time
有序 UUID 可用于新插入记录的数据库索引更好的局部性,与随机数据插入相比,这可能具有显著的性能优势。
结果包含 74 个随机位(9.25 个随机字节)。
请注意,此方法无法重现,因为其输出不仅包含随机位,还包含时间戳。
有关 UUIDv7 的详细信息,请参阅 draft-ietf-uuidrev-rfc4122bis。
单调性¶ ↑
UUIDv7 默认情况下具有毫秒精度,因此在同一毫秒内创建的多个 UUID 不会按单调递增顺序发出。要创建具有亚毫秒精度的按时间排序的 UUID,可以使用 extra_timestamp_bits
添加最多 12 位的额外时间戳。额外的时戳精度是以随机位为代价的。设置 extra_timestamp_bits: 12
提供约 244 纳秒的精度,但只有 62 个随机位(7.75 个随机字节)。
prng = Random.new Array.new(4) { prng.uuid_v7(extra_timestamp_bits: 12) } # => ["0188d4c7-13da-74f9-8b53-22a786ffdd5a", "0188d4c7-13da-753b-83a5-7fb9b2afaeea", "0188d4c7-13da-754a-88ea-ac0baeedd8db", "0188d4c7-13da-7557-83e1-7cad9cda0d8d"] # |<--- sorted --->| |<-- random --->| Array.new(4) { prng.uuid_v7(extra_timestamp_bits: 8) } # => ["0188d4c7-3333-7a95-850a-de6edb858f7e", "0188d4c7-3333-7ae8-842e-bc3a8b7d0cf9", # <- out of order "0188d4c7-3333-7ae2-995a-9f135dc44ead", # <- out of order "0188d4c7-3333-7af9-87c3-8f612edac82e"] # |<--- sorted -->||<---- random --->|
系统时钟的任何回滚都会破坏单调性。UUIDv7 基于 UTC,UTC 排除闰秒,可能会回滚时钟。为了避免这种情况,系统时钟可以与配置为使用“闰秒平滑”方法的 NTP 服务器同步。NTP 或 PTP 也将需要在分布式节点之间同步。
没有实现计数器和其他用于更强单调性保证的机制。具有更严格要求的应用程序应遵循规范的 第 6.2 节。
# File lib/random/formatter.rb, line 247 def uuid_v7(extra_timestamp_bits: 0) case (extra_timestamp_bits = Integer(extra_timestamp_bits)) when 0 # min timestamp precision ms = Process.clock_gettime(Process::CLOCK_REALTIME, :millisecond) rand = random_bytes(10) rand.setbyte(0, rand.getbyte(0) & 0x0f | 0x70) # version rand.setbyte(2, rand.getbyte(2) & 0x3f | 0x80) # variant "%08x-%04x-%s" % [ (ms & 0x0000_ffff_ffff_0000) >> 16, (ms & 0x0000_0000_0000_ffff), rand.unpack("H4H4H12").join("-") ] when 12 # max timestamp precision ms, ns = Process.clock_gettime(Process::CLOCK_REALTIME, :nanosecond) .divmod(1_000_000) extra_bits = ns * 4096 / 1_000_000 rand = random_bytes(8) rand.setbyte(0, rand.getbyte(0) & 0x3f | 0x80) # variant "%08x-%04x-7%03x-%s" % [ (ms & 0x0000_ffff_ffff_0000) >> 16, (ms & 0x0000_0000_0000_ffff), extra_bits, rand.unpack("H4H12").join("-") ] when (0..12) # the generic version is slower than the special cases above rand_a, rand_b1, rand_b2, rand_b3 = random_bytes(10).unpack("nnnN") rand_mask_bits = 12 - extra_timestamp_bits ms, ns = Process.clock_gettime(Process::CLOCK_REALTIME, :nanosecond) .divmod(1_000_000) "%08x-%04x-%04x-%04x-%04x%08x" % [ (ms & 0x0000_ffff_ffff_0000) >> 16, (ms & 0x0000_0000_0000_ffff), 0x7000 | ((ns * (1 << extra_timestamp_bits) / 1_000_000) << rand_mask_bits) | rand_a & ((1 << rand_mask_bits) - 1), 0x8000 | (rand_b1 & 0x3fff), rand_b2, rand_b3 ] else raise ArgumentError, "extra_timestamp_bits must be in 0..12" end end
私有实例方法
生成一个从字符源数组中随机抽取的字符串。
参数 source 指定要从中生成字符串的字符数组。参数 n 指定要生成的字符串的长度(以字符为单位)。
结果可能包含源数组中的任何字符。
require 'random/formatter' prng.choose([*'l'..'r'], 16) #=> "lmrqpoonmmlqlron" prng.choose([*'0'..'9'], 5) #=> "27309"
# File lib/random/formatter.rb, line 313 def choose(source, n) size = source.size m = 1 limit = size while limit * size <= 0x100000000 limit *= size m += 1 end result = ''.dup while m <= n rs = random_number(limit) is = rs.digits(size) (m-is.length).times { is << 0 } result << source.values_at(*is).join('') n -= m end if 0 < n rs = random_number(limit) is = rs.digits(size) if is.length < n (n-is.length).times { is << 0 } else is.pop while n < is.length end result.concat source.values_at(*is).join('') end result end
对 Random
的内部接口;生成 n 字节的随机数据。
# File lib/random/formatter.rb, line 295 def gen_random(n) self.bytes(n) end