Как создать случайную строку в Ruby
В настоящее время я генерирую псевдослучайную строковую строку с 8 символами для "A".. "Z":
value = ""; 8.times{value << (65 + rand(25)).chr}
но он не выглядит чистым и не может быть передан в качестве аргумента, поскольку он не является одним выражением. Чтобы получить строку смешанного случая "a".. "z" плюс "A".. "Z", я изменил ее на:
value = ""; 8.times{value << ((rand(2)==1?65:97) + rand(25)).chr}
но он выглядит как мусор.
Есть ли у кого лучший метод?
Ответы
Ответ 1
(0...8).map { (65 + rand(26)).chr }.join
Я провожу слишком много времени для игры в гольф.
(0...50).map { ('a'..'z').to_a[rand(26)] }.join
И последний, который еще более запутанный, но более гибкий и содержит меньше циклов:
o = [('a'..'z'), ('A'..'Z')].map(&:to_a).flatten
string = (0...50).map { o[rand(o.length)] }.join
Ответ 2
Почему бы не использовать SecureRandom?
require 'securerandom'
random_string = SecureRandom.hex
# outputs: 5b5cd0da3121fc53b4bc84d0c8af2e81 (i.e. 32 chars of 0..9, a..f)
SecureRandom также имеет методы для:
- base64
- random_bytes
- случайное_число
см. http://ruby-doc.org/stdlib-1.9.2/libdoc/securerandom/rdoc/SecureRandom.html
Ответ 3
Я использую это для генерации случайных URL-дружественных строк с гарантированной максимальной длиной:
rand(36**length).to_s(36)
Он генерирует случайные строки из строчных букв a-z и 0-9. Это не очень настраиваемый, но он короткий и чистый.
Ответ 4
Это решение генерирует строку легко читаемых символов для кодов активации; Я не хотел, чтобы люди путали 8 с B, 1 с I, 0 с O, L с 1 и т.д.
# Generates a random string from a set of easily readable characters
def generate_activation_code(size = 6)
charset = %w{ 2 3 4 6 7 9 A C D E F G H J K M N P Q R T V W X Y Z}
(0...size).map{ charset.to_a[rand(charset.size)] }.join
end
Ответ 5
Другие упомянули что-то подобное, но это использует безопасную для URL функцию.
require 'securerandom'
p SecureRandom.urlsafe_base64(5) #=> "UtM7aa8"
p SecureRandom.urlsafe_base64 #=> "UZLdOkzop70Ddx-IJR0ABg"
p SecureRandom.urlsafe_base64(nil, true) #=> "i0XQ-7gglIsHGV2_BNPrdQ=="
Результат может содержать A-Z, a-z, 0-9, "-" и "_". "=" также используется, если заполнение истинно.
Ответ 6
С ruby 2.5 действительно легко с SecureRandom.alphanumeric
:
len = 8
SecureRandom.alphanumeric(len)
=> "larHSsgL"
Генерирует случайные строки, содержащие A-Z, a-z и 0-9 и, следовательно, должны применяться в большинстве случаев использования. И они генерируются случайным образом, что тоже может быть выгодно.
Изменение: тест для сравнения с решением, имеющим наибольшее количество голосов:
require 'benchmark'
require 'securerandom'
len = 10
n = 100_000
Benchmark.bm(12) do |x|
x.report('SecureRandom') { n.times { SecureRandom.alphanumeric(len) } }
x.report('rand') do
o = [('a'..'z'), ('A'..'Z'), (0..9)].map(&:to_a).flatten
n.times { (0...len).map { o[rand(o.length)] }.join }
end
end
user system total real
SecureRandom 0.429442 0.002746 0.432188 ( 0.432705)
rand 0.306650 0.000716 0.307366 ( 0.307745)
Таким образом, решение rand
занимает всего около 3/4 времени SecureRandom
. Может иметь значение, если вы генерируете действительно много строк, но если вы просто время от времени создаете какую-то случайную строку, я всегда выбираю более безопасную реализацию (так как ее также проще вызывать и более явную).
Ответ 7
[*('A'..'Z')].sample(8).join
Создайте случайную 8-буквенную строку (например, NVAYXHGR)
([*('A'..'Z'),*('0'..'9')]-%w(0 1 I O)).sample(8).join
Создайте случайную 8-значную строку (например, 3PH4SWF2), исключая 0/1/I/O. Ruby 1.9
Ответ 8
Я не могу вспомнить, где я нашел это, но мне кажется, что это лучший и наименее интенсивный процесс для меня:
def random_string(length=10)
chars = 'abcdefghjkmnpqrstuvwxyzABCDEFGHJKLMNPQRSTUVWXYZ0123456789'
password = ''
length.times { password << chars[rand(chars.size)] }
password
end
Ответ 9
require 'securerandom'
SecureRandom.urlsafe_base64(9)
Ответ 10
Если вам нужна строка указанной длины, используйте:
require 'securerandom'
randomstring = SecureRandom.hex(n)
Он будет генерировать случайную строку длины 2n
, содержащую 0-9
и a-f
Ответ 11
Array.new(n){[*"0".."9"].sample}.join
где n = 8 в вашем случае.
Обобщены: Array.new(n){[*"A".."Z", *"0".."9"].sample}.join
и т.д. - из this answer
Ответ 12
require 'sha1'
srand
seed = "--#{rand(10000)}--#{Time.now}--"
Digest::SHA1.hexdigest(seed)[0,8]
Ответ 13
Ruby 1.9 +:
ALPHABET = ('a'..'z').to_a
#=> ["a", "b", "c", "d", "e", "f", "g", "h", "i", "j", "k", "l", "m", "n", "o", "p", "q", "r", "s", "t", "u", "v", "w", "x", "y", "z"]
10.times.map { ALPHABET.sample }.join
#=> "stkbssowre"
# or
10.times.inject('') { |s| s + ALPHABET.sample }
#=> "fdgvacnxhc"
Ответ 14
Вот одна строка простого кода для случайной строки длиной 8
random_string = ('0'..'z').to_a.shuffle.first(8).join
Вы также можете использовать его для случайного пароля длиной 8
random_password = ('0'..'z').to_a.shuffle.first(8).join
Надеюсь, это поможет и удивительно.
Ответ 15
Вот один простой код для случайного пароля с lenth 8
rand_password=('0'..'z').to_a.shuffle.first(8).join
Надеюсь, это поможет.
Ответ 16
Знайте: rand
является предсказуемым для злоумышленника и поэтому, вероятно, небезопасно. Вы должны обязательно использовать SecureRandom, если это для генерации паролей. Я использую что-то вроде этого:
length = 10
characters = ('A'..'Z').to_a + ('a'..'z').to_a + ('0'..'9').to_a
password = SecureRandom.random_bytes(length).each_char.map do |char|
characters[(char.ord % characters.length)]
end.join
Ответ 17
Другой метод, который я хотел бы использовать
rand(2**256).to_s(36)[0..7]
Добавьте ljust, если вы действительно параноик относительно правильной длины строки:
rand(2**256).to_s(36).ljust(8,'a')[0..7]
Ответ 18
SecureRandom.base64(15).tr('+/=lIO0', 'pqrsxyz')
Что-то из Devise
Ответ 19
Я думаю, что это хороший баланс лаконичности, ясности и легкости модификации.
characters = ('a'..'z').to_a + ('A'..'Z').to_a
# Prior to 1.9, use .choice, not .sample
(0..8).map{characters.sample}.join
Легко модифицированный
Например, включая цифры:
characters = ('a'..'z').to_a + ('A'..'Z').to_a + (0..9).to_a
Шестнадцатеричный индекс в верхнем регистре:
characters = ('A'..'F').to_a + (0..9).to_a
Для действительно впечатляющего массива символов:
characters = (32..126).to_a.pack('U*').chars.to_a
Ответ 20
Просто добавьте мои центы здесь...
def random_string(length = 8)
rand(32**length).to_s(32)
end
Ответ 21
вы можете использовать String#random
из граней Ruby Gem facets
:
https://github.com/rubyworks/facets/blob/126a619fd766bc45588cac18d09c4f1927538e33/lib/core/facets/string/random.rb
в основном это делает:
class String
def self.random(len=32, character_set = ["A".."Z", "a".."z", "0".."9"])
characters = character_set.map { |i| i.to_a }.flatten
characters_len = characters.length
(0...len).map{ characters[rand(characters_len)] }.join
end
end
Ответ 22
Дано:
chars = [*('a'..'z'),*('0'..'9')].flatten
Единое выражение, может быть передано как аргумент, позволяет дублировать символы:
Array.new(len) { chars.sample }.join
Ответ 23
Мой любимый (:A..:Z).to_a.shuffle[0,8].join
. Обратите внимание, что для перетасовки требуется Ruby > 1.9.
Ответ 24
Это решение требует внешней зависимости, но кажется красивее другого.
- Установить gem faker
-
Faker::Lorem.characters(10) # => "ang9cbhoa8"
Ответ 25
В последнее время я делал что-то подобное, чтобы генерировать 8-байтовую случайную строку из 62 символов. Символами были 0-9, a-z, A-Z. У меня был массив из них, цикл был 8 раз и выбор случайного значения из массива. Это было в приложении rails.
str = ''
8.times {|i| str << ARRAY_OF_POSSIBLE_VALUES[rand(SIZE_OF_ARRAY_OF_POSSIBLE_VALUES)] }
Странно, что я получил большое количество дубликатов. Теперь случайным образом это в значительной степени никогда не произойдет. 62 ^ 8 огромна, но из 1200 или около того кодов в db у меня было большое количество дубликатов. Я заметил, что они происходят на часовых границах друг друга. Другими словами, я мог бы увидеть дубликат в 12:12:23 и 2:12:22 или что-то в этом роде... не уверен, что время или проблема.
Этот код был создан до создания объекта activerecord. До создания записи этот код запускался и генерировал "уникальный" код. Записи в db всегда создавались надежно, но код (str в приведенной выше строке) дублировался слишком часто.
Я создал script, чтобы выполнить 100000 итераций этой выше строки с небольшой задержкой, поэтому потребуется 3-4 часа в надежде увидеть какой-то повторяющийся шаблон на почасовой основе, но ничего не увидел. Я понятия не имею, почему это происходит в моем приложении rails.
Ответ 26
''.tap {|v| 4.times { v << ('a'..'z').to_a.sample} }
Ответ 27
2 решения для случайной строки, состоящей из 3 диапазонов:
(('a'..'z').to_a + ('A'..'Z').to_a + (0..9).to_a).sample(8).join
([*(48..57),*(65..90),*(97..122)]).sample(8).collect(&:chr)*""
Один персонаж из каждого диапазона.
И если вам нужен хотя бы один символ из каждого диапазона, например, для создания случайного пароля, который состоит из одной заглавной буквы, одной строчной буквы и одной цифры, вы можете сделать что-то вроде этого:
( ('a'..'z').to_a.sample(8) + ('A'..'Z').to_a.sample(8) + (0..9).to_a.sample(8) ).shuffle.join
#=> "Kc5zOGtM0H796QgPp8u2Sxo1"
Ответ 28
Мои 2 цента:
def token(length=16)
chars = [*('A'..'Z'), *('a'..'z'), *(0..9)]
(0..length).map {chars.sample}.join
end
Ответ 29
Я просто пишу небольшой камень random_token
для генерации случайных токенов для большинства случаев использования, наслаждайтесь ~
https://github.com/sibevin/random_token
Ответ 30
С помощью этого метода вы можете пройти произвольную длину. Он устанавливается по умолчанию как 6.
def generate_random_string(length=6)
string = ""
chars = ("A".."Z").to_a
length.times do
string << chars[rand(chars.length-1)]
end
string
end