Ruby中典型的数组差示例是:
[ 1, 1, 2, 2, 3, 3, 4, 5 ] - [ 1, 2, 4 ] #=> [ 3, 3, 5 ]
如何获得以下行为的最佳方法?
[ 1, 1, 2, 2, 3, 3, 4, 5 ].subtract_once([ 1, 2, 4 ]) #=> [ 1, 2, 3, 3, 5 ]
也就是说,第二个数组中匹配的每个项的第一个实例被从第一个数组中移除。
从另一个数组中减去与之匹配数值的次数,或者使用任何Enumerable方法:
class Array
# Subtract each passed value once:
# %w(1 2 3 1).subtract_once %w(1 1 2) # => ["3"]
# [ 1, 1, 2, 2, 3, 3, 4, 5 ].subtract_once([ 1, 2, 4 ]) => [1, 2, 3, 3, 5]
# Time complexity of O(n + m)
def subtract_once(values)
counts = values.inject(Hash.new(0)) { |h, v| h[v] += 1; h }
reject { |e| counts[e] -= 1 unless counts[e].zero? }
end
减去每个唯一的值一次:
require 'set'
class Array
# Subtract each unique value once:
# %w(1 2 2).subtract_once_uniq %w(1 2 2) # => [2]
# Time complexity of O((n + m) * log m)
def subtract_once_uniq(values)
# note that set is implemented
values_set = Set.new values.to_a
reject { |e| values_set.delete(e) if values_set.include?(e) }
end
end
class Array
def subtract_once(b)
h = b.inject({}) {|memo, v|
memo[v] ||= 0; memo[v] += 1; memo
}
reject { |e| h.include?(e) && (h[e] -= 1) >= 0 }
end
end
我相信这样做可以达到我想要的效果。非常感谢 @glebm。
memo[v] ||= 0; memo[v] += 1; memo
在 reject 方法内部:
h.include?(e) && !(h[e] -= 1).zero?
- glebm[1, 2, 4].each { |x| ary.delete_at ary.index(x) }
m
([1,2,4] 的大小)很大,那可能会变得有点慢。 - glebmary
中包含[1,2,4]数组的每个元素时才有效。否则,该元素的索引为nil。内部可能是这样的:i = ary.index(x); ary.delete_at(i) if i
- Matt Sanders# remove each element of y from x exactly once
def array_difference(x, y)
ret = x.dup
y.each do |element|
if index = ret.index(element)
ret.delete_at(index)
end
end
ret
end
x = [1,2,3]
y = [3,4,5]
z = array_difference(x, y) # => [1,2]
x == [1,2,3] # => [1,2,3]