如何替换字符串中的字符

How to replace the characters in a string

我有一个方法可以用来替换字符串中的字符:

def complexity_level_two
  replacements = {
      'i' => 'eye', 'e' => 'eei',
      'a' => 'aya', 'o' => 'oha'}
  word = "Cocoa!55"
  word_arr = word.split('')
  results = []
  word_arr.each { |char|
    if replacements[char] != nil
      results.push(char.to_s.gsub!(replacements[char]))
    else
      results.push(char)
    end
  }
end

我想要的字符串输出应该是:Cohacohaa!55

然而当我运行这个方法时它不会替换字符并且只输出字符串:

C
o
c
o
a
!
5
5

我做错了什么,这个方法不会替换字符串中的正确字符以匹配 hash 中的字符,我该如何解决这个问题以获得所需的输出?

replacements = {
  'i' => 'eye', 'e' => 'eei',
  'a' => 'aya', 'o' => 'oha'}
word = "Cocoa!55"
word.gsub(Regexp.union(replacements.keys), replacements)
#⇒ "Cohacohaaya!55"

Regexp::union, String#gsub with hash.

构造方法

使用 wordsubs 参数定义方法:

def char_replacer word, subs
  word.chars.map { |c| subs.key?(c) ? subs[c] : c }.join
end

这里我们使用了三元运算符,它本质上是一个更紧凑形式的 if-else 表达式。需要注意的关键方法是 String#charsArray#mapHash#key?,请参阅 ruby-docs 了解更多信息。现在所有这些都设置好了,您可以调用您的方法,传递一个 word 字符串和您选择的 subs 散列。

示例 1

my_subs = { 'i' => 'eye', 'e' => 'eei','a' => 'aya', 'o' => 'oha' }
my_word = "Cocoa!55"

char_replacer my_word, my_subs #=> "Cohacohaaya!55"

示例 2

my_subs = { 'a' => 'p', 'e' => 'c' }
my_word = "Cocoa!55"

char_replacer my_word, my_subs #=> "Cocop!55"
replacements = { 'i' => 'eye', 'e' => 'eei', 'a' => 'aya', 'o' => 'oha' }.
  tap { |h| h.default_proc = ->(h,k) { k } }

"Cocoa!55".gsub(/./, replacements)
  #=> "Cohacohaaya!55"

参见Hash#default_proc= and Object#tap

gsub 检查字符串的每个字符。如果 replacements 将该字符作为键,则该字符将替换为 replacements 中该键的值; else(因为默认proc),字符被自身替换(即保持不变)。

另一种方法是使用 Hash#fetch:

replacements = { 'i' => 'eye', 'e' => 'eei', 'a' => 'aya', 'o' => 'oha' }

"Cocoa!55".gsub(/./) { |s| replacements.fetch(s) { |c| c } }
  #=> "Cohacohaaya!55"

其中,对于Ruby v2.2+(Object#itself首次亮相时),可以写成

"Cocoa!55".gsub(/./) { |s| replacements.fetch(s, &:itself) }

你可以尝试这样做:

my_subs = { 'i' => 'eye', 'e' => 'eei','a' => 'aya', 'o' => 'oha' }
my_word = "Cocoa!55"
my_word.split('').map{|i| my_subs[i] || i}.join

=> "Cohacohaaya!55"