假设我有一个0到9之间随机整数的列表。 我想将该列表划分为N
个子集,使得子集之和的比率等于给定的值,并且我想找出所有可能的划分。 我编写了以下代码,并使其在N = 2
的情况下工作。
import random
import itertools
#lst = [random.randrange(10) for _ in range(20)]
lst = [2, 0, 1, 7, 2, 4, 9, 7, 6, 0, 5, 4, 7, 4, 5, 0, 4, 5, 2, 3]
def partition_sum_with_ratio(numbers, ratios):
target1 = round(int(sum(numbers) * ratios[0] / (ratios[0] + ratios[1])))
target2 = sum(numbers) - target1
p1 = [seq for i in range(len(numbers), 0, -1) for seq in
itertools.combinations(numbers, i) if sum(seq) == target1
and sum([s for s in numbers if s not in seq]) == target2]
p2 = [tuple(n for n in numbers if n not in seq) for seq in p1]
return list(zip(p1, p2))
partitions = partition_sum_with_ratios(lst, ratios=[4, 3])
print(partitions[0])
输出:
((2, 0, 1, 2, 4, 6, 0, 5, 4, 4, 5, 0, 4, 5, 2), (7, 9, 7, 7, 3))
如果您计算每个子集的和,您会发现比例是44:33 = 4:3,正好是输入值。然而,我希望该函数适用于任意数量的子集。例如,我希望...
partition_sum_with_ratio(lst, ratios=[4, 3, 3])
返回类似于以下内容的结果
((2, 0, 1, 2, 4, 6, 0, 5, 4, 4, 3), (5, 0, 4, 5, 2, 7), (9, 7, 7))
我已经思考这个问题一个月了,发现这个问题非常难。我的结论是这个问题只能通过递归来解决。我想知道是否有任何相对快速的算法可用于此。有什么建议吗?
ratio = [4, 3]
表示两个子集的和的比例为 4:3。 - Shaun Han