如何检查数组是否有重复项?

Ski*_*zit 68 ruby arrays

我有一个数组A.我想检查它是否包含重复值.我该怎么办?

sep*_*p2k 127

只需调用uniq它(它返回一个没有重复的新数组)并查看uniqed数组是否具有比原始数组少的元素:

if a.uniq.length == a.length
  puts "a does not contain duplicates"
else
  puts "a does contain duplicates"
end
Run Code Online (Sandbox Code Playgroud)

请注意,数组中的对象需要响应hash并且eql?有意义uniq才能正常工作.

  • @phoffer,检查大小更快. (15认同)
  • 另外,如果没有找到重复项,`uniq!`将返回`nil`,改变self以删除重复项.有很多可用的数组方法:http://ruby-doc.org/core/classes/Array.html (10认同)
  • @ sepp2k出于好奇,为什么你有'a.uniq.length == a.length`而不只是'a.uniq == a`? (5认同)
  • 当然,如果他们没有以有意义的方式回应"eql?"那么"复制"甚至**是什么意思**?一旦你定义了`eql?`,`hash`应该与它一致. (2认同)

jmo*_*iro 35

为了找到重复的元素,我使用这种方法(使用Ruby 1.9.3):

array = [1, 2, 1, 3, 5, 4, 5, 5]
=> [1, 2, 1, 3, 5, 4, 5, 5]
dup = array.select{|element| array.count(element) > 1 }
=> [1, 1, 5, 5, 5]
dup.uniq
=> [1, 5]
Run Code Online (Sandbox Code Playgroud)

  • select {array.count}是一个嵌套循环,你正在做一个O(n ^ 2)复杂算法,可以在O(n)中完成. (4认同)
  • 排序并删除n log n的连续符号 (3认同)

Ben*_*ier 10

如果要返回重复项,可以执行以下操作:

dups = [1,1,1,2,2,3].group_by{|e| e}.keep_if{|_, e| e.length > 1}
# => {1=>[1, 1, 1], 2=>[2, 2]}
Run Code Online (Sandbox Code Playgroud)

如果你只想要值:

dups.keys
# => [1, 2]
Run Code Online (Sandbox Code Playgroud)

如果你想要重复的数量:

dups.map{|k, v| {k => v.length}}
# => [{1=>3}, {2=>2}]
Run Code Online (Sandbox Code Playgroud)


fak*_*eft 5

如果不止一次使用它,可能想要monkeypatch Array:

class Array
  def uniq?
    self.length == self.uniq.length
  end
end
Run Code Online (Sandbox Code Playgroud)

然后:

irb(main):018:0> [1,2].uniq?
=> true
irb(main):019:0> [2,2].uniq?
=> false
Run Code Online (Sandbox Code Playgroud)

  • 我会避免猴子修补来包裹单衬。 (19认同)
  • 一个(主要?)原因是,如果您在最终被其他人(gems)使用的代码中执行此操作,他们会拉入您的monkeypatches,这可能会导致意外行为。所以我想我应该添加一个警告,这应该只用于你自己使用的代码中,并且在你能记住的地方(monkey_patch.rb 文件?),所以你不必寻找太多更改默认行为的代码。 (4认同)