我想从两个哈希数组得到union / intersect / difference,例如:
array1 = [{:name =>'Guy1', :age => 45},{:name =>'Guy2', :age => 45}]
array2 = [{:name =>'Guy1', :age => 45},{:name =>'Guy3', :age => 45}]
...
p array1 - array2
=> [{:name=>"Guy2", :age=>45}]
p array2 - array1
=> [{:name=>"Guy3", :age=>45}]
p array1 | array2
=> [{:name=>"Guy1", :age=>45}, {:name=>"Guy2", :age=>45}, {:name=>"Guy3", :age=>45}]
然而,当我只想根据名称进行比较并忽略年龄而不需要从哈希中删除它们时,例如:
array1 = [{:name =>'Guy1', :age => 45},{:name =>'Guy2', :age => 45}]
array2 = [{:name =>'Guy1', :age => 46},{:name =>'Guy3', :age => 45}]
在这种情况下,我没有得到我想要的结果b / c年龄不同。
array1 - array2
=> [{:name=>"Guy1", :age=>45}, {:name=>"Guy2", :age=>45}]
array2 - array1
=> [{:name=>"Guy1", :age=>46}, {:name=>"Guy3", :age=>45}]
array1 | array2
=> [{:name=>"Guy1", :age=>45}, {:name=>"Guy2", :age=>45}, {:name=>"Guy1", :age=>46}, {:name=>"Guy3", :age=>45}]
有没有办法获得union / intersect / difference并忽略年龄键?
编辑:更好的例子:
array1 = [{:name =>'Guy1', :age => 45},{:name =>'Guy2', :age => 45}]
array2 = [{:name =>'Guy1'},{:name =>'Guy3'}]
p array1 - array2
p array2 - array1
p array1 | array2
p array1 & array2
提前感谢您的帮助!
答案 0 :(得分:6)
这是获得工会的快速而肮脏的方式:
(array1 + array2).uniq{|a| a[:name]}
但是,我建议您创建自己的Hash
子类,这样您就可以安全地覆盖eql?
,因为Cary Swoveland指出了类似集合的运算符所依赖的内容。请注意,您还需要将hash
方法限制为仅在名称字段上提供散列函数。
class Guy < Hash
def eql?(other_hash)
self[:name] == other_hash[:name]
end
def hash
self[:name].hash
end
end
然后这些Guy
对象将在所有设置操作中起作用:
array1 = [ Guy[name:'Guy1', age: 45], Guy[name:'Guy2', age: 45] ]
array2 = [ Guy[name:'Guy1', age: 46], Guy[name:'Guy3', age: 45] ]
array1 - array2
#=> [{:name=>"Guy2", :age=>45}]
array2 - array1
#=> [{:name=>"Guy3", :age=>45}]
array1 | array2
#=> [{:name=>"Guy1", :age=>45}, {:name=>"Guy2", :age=>45}, {:name=>"Guy3", :age=>
45}]
array1 & array2
#=> [{:name=>"Guy1", :age=>45}]
答案 1 :(得分:0)
区别对待:
diff_arr = array1.map{|a| a[:name]} - array2.map{|a| a[:name]}
diff_arr.map{|a| array1.map{|s| s if s[:name] == a }}.flatten.compact
交叉口
intersec_arr = array1.map{|a| a[:name]} & array2.map{|a| a[:name]}
intersec_ar.map{|a| array1.map{|s| s if s[:name] == a }}.flatten.compact
答案 2 :(得分:0)
引用的所有三个类Array
方法都使用Hash#eql?来比较两个哈希值的两个元素。 (Hash#eql?
检查两个哈希值的哈希码是否相等。)因此,我们只需要(暂时)重新定义Hash#eql?
。
array1 = [{:name =>'Guy1', :age => 45}, {:name =>'Guy2', :age => 45}]
array2 = [{:name =>'Guy1'}, {:name =>'Guy3'}]
class Hash
alias old_eql? eql?
def eql?(h) self[:name] == h[:name] end
end
array1 - array2
#=> [{:name=>"Guy2", :age=>45}]
array2 - array1
#=> [{:name=>"Guy3"}]
array1 | array2
#=> [{:name=>"Guy1", :age=>45}, {:name=>"Guy2", :age=>45}, {:name=>"Guy3"}]
array1 & array2
#=> [{:name=>"Guy1", :age=>45}]
class Hash
alias eql? old_eql? # Restore eql?
undef_method :old_eql?
end
请注意,这是self
必须明确的少数几种情况之一。如果我们写道:
[:name] == h[:name]
而不是:
self[:name] == h[:name]
Ruby会假设我们正在将数组[:name]
与h[:name]
进行比较。