What would be the best way to merge arrays nested in an array that shares at least an element ? Here's an example:
合并嵌套在至少共享一个元素的数组的最佳方式是什么?这里有一个例子:
some_method([[1, 2], [2, 3], [4, 5]])
#=> [[1, 2, 3], [4, 5]]
some_method([[1, 2], [2, 3], [3, 4], [5,6]])
#=> [[1, 2, 3, 4], [5, 6]]
4 个解决方案
#1
3
This would work:
这将工作:
def some_method(arrays)
h = Hash.new { |h, k| h[k] = [] }
arrays.each do |array|
tmp = h.values_at(*array).push(array).inject(:|)
tmp.each { |k| h[k] = tmp }
end
h.values | h.values
end
Examples:
例子:
some_method([[1, 2], [2, 3], [4, 5]]) #=> [[1, 2, 3], [4, 5]]
some_method([[1, 2], [2, 3], [3, 4], [5, 6]]) #=> [[1, 2, 3, 4], [5, 6]]
some_method([[1, 3], [3, 4], [2, 5], [4, 5]]) #=> [[1, 3, 4, 2, 5]]
I'm using a hash h
to store the array that correspond to a given element. The hash returns []
if a key doesn't exist.
我使用哈希h来存储对应于给定元素的数组。如果键不存在,散列返回[]。
After inserting [1, 2]
, the hash looks like this:
插入[1,2]之后,哈希看起来如下:
{
1 => [1, 2],
2 => [1, 2]
}
When inserting [2, 3]
, the arrays for 2
and 3
are fetched via:
当插入[2,3]时,通过以下方法获取2和3的数组:
h.values_at(2, 3)
#=> [[1, 2], []]
then [2, 3]
itself is added:
然后加上[2,3]本身:
h.values_at(2, 3).push([2, 3])
#=> [[1, 2], [], [2, 3]]
and everything is |
-ed:
和一切|连接:
h.values_at(2, 3).push([2, 3]).inject(:|)
#=> [1, 2, 3]
This result is stored in tmp
. It becomes the new value for the contained keys:
这个结果存储在tmp中。它成为所包含键的新值:
tmp.each { |k| h[k] = tmp }
Which is equivalent to:
相当于:
h[1] = tmp
h[2] = tmp
h[3] = tmp
Afterwards, h
looks like this:
之后h是这样的:
{
1 => [1, 2, 3],
2 => [1, 2, 3],
3 => [1, 2, 3]
}
At the end, the distinct values are returned via h.values | h.values
.
最后,不同的值通过h返回。值| h.values。
#2
2
arr = [[1, 2], [2, 3], [3, 4], [5, 6]]
arr.map(&:dup).sort.each_with_object([]) do |a, memo|
(idx = memo.index { |m| !(m & a).empty? }) ? memo[idx] |= a : memo << a
end
#⇒ [[1, 2, 3, 4], [5, 6]]
or, more expressive:
或者,更表达:
arr.map(&:dup).sort.each_with_object([]) do |a, memo|
(memo.detect { |m| !(m & a).empty? } << a).
flatten!.uniq! rescue memo << a
end
the most precise solution, that works for any permutations, but consumes more time:
最精确的解决方案,适用于任何排列,但消耗的时间更多:
loop.inject(arr.map(&:dup)) do |acc|
result = (acc.each_with_object([]) do |a, memo|
(idx = memo.index { |m| !(m & a).empty? }) ? memo[idx] |= a : memo << a
end)
result == acc ? (break result) : result
end
#3
2
Here's a very simple approach. The steps are as follows.
这里有一个非常简单的方法。步骤如下。
-
Beginning with an array
a = arr.map(&:uniq)
,arr
being the initial array of arrays, look for two arrays ofa
that share an element, among all combinations of two arrays ofa
. If none are found, returna
(fini!); else go to step 2.从数组a = arr.map(&:uniq)开始,arr是数组的初始数组,在两个数组a的所有组合中查找共享一个元素的a的两个数组。否则到第二步。
-
If
a[i]
anda[j]
are found to contain a common element,a[i]
becomesa[i].concat(a[j]).uniq
anda[j]
is deleted.如果a[i]和a[j]包含一个共同的元素,那么a[i]就变成了一个[i].concat(a[j])。uniq和a[j]被删除。
-
Repeat #1.
重复# 1。
def group_unique(arr)
a = arr.map(&:uniq)
loop do
(_,i),(_,j) = a.each_with_index.to_a.combination(2).find {|(a,_),(b,_)|(a&b).any?}
return a if i.nil?
a[i] = a[i].concat(a.delete_at(j)).uniq
end
end
arr = [[1,2], [5,6], [2,3], [4,5], [4,1], [7,8], [11,13], [8,10]]
group_unique(arr)
#=> [[1, 2, 3, 4, 5, 6], [7, 8, 10], [11, 13]]
#4
1
It's a little verbose, but here is a recursive method that solves the problem properly:
这有点啰嗦,但这里有一个递归方法可以正确地解决问题:
def merge_shared_elements(list)
changed = false
result = list.each_with_object([]) do |item, new_list|
if existing_item = new_list.find {|new_item| !(new_item & item).empty?}
existing_item.concat(item).uniq!
changed = true
else
new_list << item
end
end
changed ? merge_shared_elements(result) : result
end
This will keep re-iterating through the list, so the order of inputs is irrelevant.
这将继续遍历列表,因此输入的顺序无关紧要。
#1
3
This would work:
这将工作:
def some_method(arrays)
h = Hash.new { |h, k| h[k] = [] }
arrays.each do |array|
tmp = h.values_at(*array).push(array).inject(:|)
tmp.each { |k| h[k] = tmp }
end
h.values | h.values
end
Examples:
例子:
some_method([[1, 2], [2, 3], [4, 5]]) #=> [[1, 2, 3], [4, 5]]
some_method([[1, 2], [2, 3], [3, 4], [5, 6]]) #=> [[1, 2, 3, 4], [5, 6]]
some_method([[1, 3], [3, 4], [2, 5], [4, 5]]) #=> [[1, 3, 4, 2, 5]]
I'm using a hash h
to store the array that correspond to a given element. The hash returns []
if a key doesn't exist.
我使用哈希h来存储对应于给定元素的数组。如果键不存在,散列返回[]。
After inserting [1, 2]
, the hash looks like this:
插入[1,2]之后,哈希看起来如下:
{
1 => [1, 2],
2 => [1, 2]
}
When inserting [2, 3]
, the arrays for 2
and 3
are fetched via:
当插入[2,3]时,通过以下方法获取2和3的数组:
h.values_at(2, 3)
#=> [[1, 2], []]
then [2, 3]
itself is added:
然后加上[2,3]本身:
h.values_at(2, 3).push([2, 3])
#=> [[1, 2], [], [2, 3]]
and everything is |
-ed:
和一切|连接:
h.values_at(2, 3).push([2, 3]).inject(:|)
#=> [1, 2, 3]
This result is stored in tmp
. It becomes the new value for the contained keys:
这个结果存储在tmp中。它成为所包含键的新值:
tmp.each { |k| h[k] = tmp }
Which is equivalent to:
相当于:
h[1] = tmp
h[2] = tmp
h[3] = tmp
Afterwards, h
looks like this:
之后h是这样的:
{
1 => [1, 2, 3],
2 => [1, 2, 3],
3 => [1, 2, 3]
}
At the end, the distinct values are returned via h.values | h.values
.
最后,不同的值通过h返回。值| h.values。
#2
2
arr = [[1, 2], [2, 3], [3, 4], [5, 6]]
arr.map(&:dup).sort.each_with_object([]) do |a, memo|
(idx = memo.index { |m| !(m & a).empty? }) ? memo[idx] |= a : memo << a
end
#⇒ [[1, 2, 3, 4], [5, 6]]
or, more expressive:
或者,更表达:
arr.map(&:dup).sort.each_with_object([]) do |a, memo|
(memo.detect { |m| !(m & a).empty? } << a).
flatten!.uniq! rescue memo << a
end
the most precise solution, that works for any permutations, but consumes more time:
最精确的解决方案,适用于任何排列,但消耗的时间更多:
loop.inject(arr.map(&:dup)) do |acc|
result = (acc.each_with_object([]) do |a, memo|
(idx = memo.index { |m| !(m & a).empty? }) ? memo[idx] |= a : memo << a
end)
result == acc ? (break result) : result
end
#3
2
Here's a very simple approach. The steps are as follows.
这里有一个非常简单的方法。步骤如下。
-
Beginning with an array
a = arr.map(&:uniq)
,arr
being the initial array of arrays, look for two arrays ofa
that share an element, among all combinations of two arrays ofa
. If none are found, returna
(fini!); else go to step 2.从数组a = arr.map(&:uniq)开始,arr是数组的初始数组,在两个数组a的所有组合中查找共享一个元素的a的两个数组。否则到第二步。
-
If
a[i]
anda[j]
are found to contain a common element,a[i]
becomesa[i].concat(a[j]).uniq
anda[j]
is deleted.如果a[i]和a[j]包含一个共同的元素,那么a[i]就变成了一个[i].concat(a[j])。uniq和a[j]被删除。
-
Repeat #1.
重复# 1。
def group_unique(arr)
a = arr.map(&:uniq)
loop do
(_,i),(_,j) = a.each_with_index.to_a.combination(2).find {|(a,_),(b,_)|(a&b).any?}
return a if i.nil?
a[i] = a[i].concat(a.delete_at(j)).uniq
end
end
arr = [[1,2], [5,6], [2,3], [4,5], [4,1], [7,8], [11,13], [8,10]]
group_unique(arr)
#=> [[1, 2, 3, 4, 5, 6], [7, 8, 10], [11, 13]]
#4
1
It's a little verbose, but here is a recursive method that solves the problem properly:
这有点啰嗦,但这里有一个递归方法可以正确地解决问题:
def merge_shared_elements(list)
changed = false
result = list.each_with_object([]) do |item, new_list|
if existing_item = new_list.find {|new_item| !(new_item & item).empty?}
existing_item.concat(item).uniq!
changed = true
else
new_list << item
end
end
changed ? merge_shared_elements(result) : result
end
This will keep re-iterating through the list, so the order of inputs is irrelevant.
这将继续遍历列表,因此输入的顺序无关紧要。