Как я могу проверить, содержит ли массив Ruby одно из нескольких значений? - PullRequest
29 голосов
/ 09 апреля 2010

У меня есть два массива Ruby, и мне нужно посмотреть, имеют ли они какие-либо общие значения. Я мог бы просто просмотреть каждое из значений в одном массиве и включить? () Для другого, но я уверен, что есть лучший способ. Что это? (Оба массива содержат строки.)

Спасибо.

Ответы [ 4 ]

81 голосов
/ 09 апреля 2010

Установить пересечение их:

a1 & a2

Вот пример:

> a1 = [ 'foo', 'bar' ]
> a2 = [ 'bar', 'baz' ]
> a1 & a2
=> ["bar"]
> !(a1 & a2).empty? # Returns true if there are any elements in common
=> true
7 голосов
/ 09 апреля 2010

Есть что-нибудь общее? Вы можете использовать оператор пересечения: &

[ 1, 1, 3, 5 ] & [ 1, 2, 3 ]   #=> [ 1, 3 ]

Однако если вы ищете полное пересечение (с дубликатами), проблема более сложная, здесь уже переполнение стека: Как вернуть пересечение массива Ruby с дублирующимися элементами? (проблема с биграммами в коэффициентах костей)

Или быстрый фрагмент , который определяет "real_intersection" и проверяет следующий тест

class ArrayIntersectionTests < Test::Unit::TestCase    
  def test_real_array_intersection
    assert_equal [2], [2, 2, 2, 3, 7, 13, 49] & [2, 2, 2, 5, 11, 107]
    assert_equal [2, 2, 2], [2, 2, 2, 3, 7, 13, 49].real_intersection([2, 2, 2, 5, 11, 107])
    assert_equal ['a', 'c'], ['a', 'b', 'a', 'c'] & ['a', 'c', 'a', 'd']
    assert_equal ['a', 'a', 'c'], ['a', 'b', 'a', 'c'].real_intersection(['a', 'c', 'a', 'd'])
  end
end
4 голосов
/ 09 ноября 2015

Использование пересечения выглядит красиво, но неэффективно. Я бы использовал "любой?" на первом массиве (так что итерация останавливается, когда один из элементов находится во втором массиве). Кроме того, использование Set во втором массиве сделает проверку членства быстрой. i.e.:

a = [:a, :b, :c, :d]
b = Set.new([:c, :d, :e, :f])
c = [:a, :b, :g, :h]

# Do a and b have at least a common value?
a.any? {|item| b.include? item}
# true

# Do c and b have at least a common value?
c.any? {|item| b.include? item}
#false
0 голосов
/ 18 апреля 2017

Попробуйте это

a1 = [ 'foo', 'bar' ] 
a2 = [ 'bar', 'baz' ]
a1-a2 != a1
true
...