Ruby中奇怪的反斜杠替换

时间:2009-10-09 06:59:14

标签: ruby regex backslash

我不明白这个Ruby代码:

>> puts '\\ <- single backslash'
# \ <- single backslash

>> puts '\\ <- 2x a, because 2 backslashes get replaced'.sub(/\\/, 'aa')
# aa <- 2x a, because two backslashes get replaced

到目前为止,一切如预期。但是,如果我们使用/\\/搜索1,并使用'\\\\'编码的2替换,为什么我们会这样做:

>> puts '\\ <- only 1 ... replace 1 with 2'.sub(/\\/, '\\\\')
# \ <- only 1 backslash, even though we replace 1 with 2

然后,当我们用'\\\\\\'编码3时,我们只得到2:

>> puts '\\ <- only 2 ... 1 with 3'.sub(/\\/, '\\\\\\')
# \\ <- 2 backslashes, even though we replace 1 with 3

任何人都能理解为什么在替换字符串中吞下反斜杠?这发生在1.8和1.9。

5 个答案:

答案 0 :(得分:62)

快速回答

如果你想回避所有这些混乱,使用更少混乱的块语法。下面是一个用2个反斜杠替换每个反斜杠的示例:

"some\\path".gsub('\\') { '\\\\' }

令人毛骨悚然的细节

问题在于,当使用sub(和gsub)时,如果没有块,ruby会在替换参数中解释特殊字符序列。不幸的是,sub使用反斜杠作为这些的转义字符:

\& (the entire regex)
\+ (the last group)
\` (pre-match string)
\' (post-match string)
\0 (same as \&)
\1 (first captured group)
\2 (second captured group)
\\ (a backslash)

与任何转义一样,这会产生明显的问题。如果要在输出字符串中包含上述序列之一(例如\1)的文字值,则必须将其转义。因此,要获得Hello \1,您需要将替换字符串设为Hello \\1。要在Ruby中将其表示为字符串文字,您必须再次转义这些反斜杠:"Hello \\\\1"

所以,有两个不同的转义。第一个采用字符串文字并创建内部字符串值。第二个获取内部字符串值,并用匹配数据替换上面的序列。

如果反斜杠后面没有与上述序列之一匹配的字符,则反斜杠(以及后面的字符)将不加改变地通过。这也会影响字符串末尾的反斜杠 - 它将不加改变地传递。在rubinius代码中最容易看到这个逻辑;只需在String class中查找to_sub_replacement方法即可。

以下是一些示例,了解String#sub如何解析替换字符串:

  • 1反斜杠 \(其字符串文字为"\\"

    传递不变,因为反斜杠位于字符串的末尾并且后面没有字符。

    结果: \

  • 2反斜杠 \\(其字符串文字为"\\\\"

    这对反斜杠与转义的反斜杠序列匹配(参见上面的\\)并转换为单个反斜杠。

    结果: \

  • 3个反斜杠 \\\(字符串文字为"\\\\\\"

    前两个反斜杠与\\序列匹配,并转换为单个反斜杠。然后最后的反斜杠位于字符串的末尾,因此它不会改变。

    结果: \\

  • 4个反斜杠 \\\\(字符串文字为"\\\\\\\\"

    两对反斜杠分别与\\序列匹配,并转换为单个反斜杠。

    结果: \\

  • 中间带有字符的2个反斜杠 \a\(其字符串文字为"\\a\\"

    \a与任何转义序列都不匹配,因此允许它保持不变。尾随反斜杠也允许通过。

    结果: \a\

    注意:可以从\\a\\获取相同的结果(使用文字字符串:"\\\\a\\\\"

事后看来,如果String#sub使用了不同的转义字符,这可能不那么令人困惑。然后就不需要双重逃避所有的反斜杠了。

答案 1 :(得分:18)

这是一个问题,因为反斜杠(\)用作Regexps和Strings的转义字符。你可以使用特殊变量\&amp;减少gsub替换字符串中的反斜杠数。

foo.gsub(/\\/,'\&\&\&') #for some string foo replace each \ with \\\
编辑:我应该提到\&amp;的价值来自Regexp匹配,在这种情况下是一个反斜杠。

另外,我认为有一种特殊的方法来创建一个禁用转义字符的字符串,但显然不是。这些都不会产生两个斜杠:

puts "\\"
puts '\\'
puts %q{\\}
puts %Q{\\}
puts """\\"""
puts '''\\'''
puts <<EOF
\\
EOF  

答案 2 :(得分:4)

argh,在我输入所有内容之后,我意识到\用于指代替换字符串中的组。我想这意味着您需要在替换字符串中使用文字\\来替换\。要获得文字\\,您需要四个\,所以要将两个替换为实际需要八个(!)。

# Double every occurrence of \. There's eight backslashes on the right there!
>> puts '\\'.sub(/\\/, '\\\\\\\\')

我错过了什么?更有效的方式?

答案 3 :(得分:3)

清除了对作者第二行代码的一点混淆。

你说:

>> puts '\\ <- 2x a, because 2 backslashes get replaced'.sub(/\\/, 'aa')
# aa <- 2x a, because two backslashes get replaced

这里没有替换2个反斜杠。您正在用两个a('aa')替换 1个转义反斜杠。也就是说,如果您使用.sub(/\\/, 'a'),则只会看到一个'a'

'\\'.sub(/\\/, 'anything') #=> anything

答案 4 :(得分:2)

实际上,镐书提到了这个确切的问题。这是另一种选择(来自最新版本的第130页)

str = 'a\b\c'               # => "a\b\c"
str.gsub(/\\/) { '\\\\' }   # => "a\\b\\c"