Поиск общей строки в массиве строк (ruby) - PullRequest
6 голосов
/ 29 января 2010

Учитывая, что у меня есть массив из 3 строк:

["Extra tv in bedroom",
 "Extra tv in living room",
 "Extra tv outside the shop"]

Как найти самую длинную строку из всех общих?

Ответы [ 6 ]

15 голосов
/ 29 января 2010

Вот рубиновый способ сделать это. Вам следует использовать более продвинутый алгоритм, если у вас есть несколько строк или они очень длинные:

def longest_common_substr(strings)
  shortest = strings.min_by &:length
  maxlen = shortest.length
  maxlen.downto(0) do |len|
    0.upto(maxlen - len) do |start|
      substr = shortest[start,len]
      return substr if strings.all?{|str| str.include? substr }
    end
  end
end

puts longest_common_substr(["Extra tv in bedroom",
                            "Extra tv in living room",
                            "Extra tv outside the shop"])
2 голосов
/ 29 января 2010

Если вы хотите найти начало всех строк:

Источник

def substr( a )
    return "" unless (a.length > 0)
    result = 0
    (0 ... a.first.length).each do |k|
        all_matched = true
        character = a.first[k]
        a.each{ |str| all_matched &= (character == str[k]) }
        break unless all_matched
        result+=1
    end
    a.first.slice(0,result)
end

Test

input = ["Extra tv in bedroom",
 "Extra tv in living room",
 "Extra tv outside the shop"]

puts substr( input ) + "."

выход

Extra tv .
2 голосов
/ 29 января 2010

Эта статья в Википедии объясняет два алгоритма, которые можно использовать для решения этой проблемы.

1 голос
/ 03 февраля 2010

Также только для начала строк.

def longest_subsequence array
  array.sort!
  first = array[0].split(//)
  last = array[-1].split(//)
  length = (first.size > last.size) ? last.size : first.size
  sequence = ""
  index = 0
  while (first[index] == last[index]) && (index < length)
    sequence << first[index]
    index += 1
  end
  sequence
end

Но я думаю, что должен быть способ легко сравнить начало двух строк для соответствующей подстроки - я просто не могу об этом думать прямо сейчас!

0 голосов
/ 01 июля 2010

Не знаю, полезен ли ответ, но вот решение, вдохновленное кодами @mckeed и @ lins314159.

def longest_common_substr(strings)
    longest_substring = strings.map{|s| s.split}.max_by &:length
    longest_substring.inject do |target_str, token|
        r = Regexp.new("^#{target_str.nil? ? token : "#{target_str} #{token}".strip}")
        target_str = "#{target_str} #{token}".strip if strings.all? {|string| string =~ r}
        target_str
    end
end

puts longest_common_substr(["Extra tv and mat in bedroom",
                            "Extra tv and chair with view in living room",
                            "Extra tv and carpet outside the shop"])
0 голосов
/ 29 января 2010

Не думаю, что это особенно хорошо масштабируется.

def longest_substr(text)
    if (text.length == 0)
        return ""
    elseIf (text.length == 1)
        return text[0]
    end
    longest = text.inject(text[0].length) {|min, s| min < s.length ? min : s.length}
    (1 .. longest).to_a.reverse.each do |l|
        (0 .. text[0].length - l).each do |offset|
            str = text[0].slice(offset, l)
            matched = (1 .. text.length - 1).inject(true) {|matched, i| matched && text[i].index(str) != nil}
            if (matched)
                return str
            end
        end
    end

    return ""
end

puts longest_substr(["Alice's Extra tv in bedroom",
    "Bob's Extra tv in living room",
    "My Extra tv outside the shop"])
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...