How to replace the last occurrence of a substring in ruby?

前端 未结 10 1125
南笙
南笙 2021-02-03 20:08

I want to replace the last occurrence of a substring in Ruby. What\'s the easiest way? For example, in abc123abc123, I want to replace the last abc

相关标签:
10条回答
  • 2021-02-03 20:48

    simple and efficient:

    s = "abc123abc123abc"
    p = "123"
    s.slice!(s.rindex(p), p.size)
    s == "abc123abcabc"
    
    0 讨论(0)
  • 2021-02-03 20:49
    "abc123abc123".gsub(/(.*(abc.*)*)(abc)(.*)/, '\1ABC\4')
    #=> "abc123ABC123"
    

    But probably there is a better way...

    Edit:

    ...which Chris kindly provided in the comment below.

    So, as * is a greedy operator, the following is enough:

    "abc123abc123".gsub(/(.*)(abc)(.*)/, '\1ABC\3')
    #=> "abc123ABC123"
    

    Edit2:

    There is also a solution which neatly illustrates parallel array assignment in Ruby:

    *a, b = "abc123abc123".split('abc', -1)
    a.join('abc')+'ABC'+b
    #=> "abc123ABC123"
    
    0 讨论(0)
  • 2021-02-03 20:58
    .gsub /abc(?=[^abc]*$)/, 'ABC'
    

    Matches a "abc" and then asserts ((?=) is positive lookahead) that no other characters up to the end of the string are "abc".

    0 讨论(0)
  • 2021-02-03 20:59

    How about

    new_str = old_str.reverse.sub(pattern.reverse, replacement.reverse).reverse
    

    For instance:

    irb(main):001:0> old_str = "abc123abc123"
    => "abc123abc123"
    irb(main):002:0> pattern="abc"
    => "abc"
    irb(main):003:0> replacement="ABC"
    => "ABC"
    irb(main):004:0> new_str = old_str.reverse.sub(pattern.reverse, replacement.reverse).reverse
    => "abc123ABC123"
    
    0 讨论(0)
  • 2021-02-03 21:03

    Here's another possible solution:

    >> s = "abc123abc123"
    => "abc123abc123"
    
    >> s[s.rindex('abc')...(s.rindex('abc') + 'abc'.length)] = "ABC"
    => "ABC"
    
    >> s
    => "abc123ABC123"
    
    0 讨论(0)
  • 2021-02-03 21:03

    When searching in huge streams of data, using reverse will definitively* lead to performance issues. I use string.rpartition*:

    sub_or_pattern = "!"
    replacement = "?"
    string = "hello!hello!hello"
    
    array_of_pieces = string.rpartition sub_or_pattern
    ( array_of_pieces[(array_of_pieces.find_index sub_or_pattern)] =  replacement ) rescue nil
    p array_of_pieces.join
    # "hello!hello?hello"
    

    The same code must work with a string with no occurrences of sub_or_pattern:

    string = "hello_hello_hello"
    # ...
    # "hello_hello_hello"
    

    *rpartition uses rb_str_subseq() internally. I didn't check if that function returns a copy of the string, but I think it preserves the chunk of memory used by that part of the string. reverse uses rb_enc_cr_str_copy_for_substr(), which suggests that copies are done all the time -- although maybe in the future a smarter String class may be implemented (having a flag reversed set to true, and having all of its functions operating backwards when that is set), as of now, it is inefficient.

    Moreover, Regex patterns can't be simply reversed. The question only asks for replacing the last occurrence of a sub-string, so, that's OK, but readers in the need of something more robust won't benefit from the most voted answer (as of this writing)

    0 讨论(0)
提交回复
热议问题