2023-10-15 19:56:02 +00:00
|
|
|
def subset_combinations(elements: list[int], n: int) -> list:
|
2020-05-22 06:10:11 +00:00
|
|
|
"""
|
2023-10-15 19:56:02 +00:00
|
|
|
Compute n-element combinations from a given list using dynamic programming.
|
2024-12-30 11:52:03 +00:00
|
|
|
|
2023-10-15 19:56:02 +00:00
|
|
|
Args:
|
2024-12-30 11:52:03 +00:00
|
|
|
* `elements`: The list of elements from which combinations will be generated.
|
|
|
|
* `n`: The number of elements in each combination.
|
|
|
|
|
2023-10-15 19:56:02 +00:00
|
|
|
Returns:
|
2024-12-30 11:52:03 +00:00
|
|
|
A list of tuples, each representing a combination of `n` elements.
|
|
|
|
|
|
|
|
>>> subset_combinations(elements=[10, 20, 30, 40], n=2)
|
|
|
|
[(10, 20), (10, 30), (10, 40), (20, 30), (20, 40), (30, 40)]
|
|
|
|
>>> subset_combinations(elements=[1, 2, 3], n=1)
|
|
|
|
[(1,), (2,), (3,)]
|
|
|
|
>>> subset_combinations(elements=[1, 2, 3], n=3)
|
|
|
|
[(1, 2, 3)]
|
|
|
|
>>> subset_combinations(elements=[42], n=1)
|
|
|
|
[(42,)]
|
|
|
|
>>> subset_combinations(elements=[6, 7, 8, 9], n=4)
|
|
|
|
[(6, 7, 8, 9)]
|
|
|
|
>>> subset_combinations(elements=[10, 20, 30, 40, 50], n=0)
|
|
|
|
[()]
|
|
|
|
>>> subset_combinations(elements=[1, 2, 3, 4], n=2)
|
|
|
|
[(1, 2), (1, 3), (1, 4), (2, 3), (2, 4), (3, 4)]
|
|
|
|
>>> subset_combinations(elements=[1, 'apple', 3.14], n=2)
|
|
|
|
[(1, 'apple'), (1, 3.14), ('apple', 3.14)]
|
|
|
|
>>> subset_combinations(elements=['single'], n=0)
|
|
|
|
[()]
|
|
|
|
>>> subset_combinations(elements=[], n=9)
|
|
|
|
[]
|
|
|
|
>>> from itertools import combinations
|
|
|
|
>>> all(subset_combinations(items, n) == list(combinations(items, n))
|
|
|
|
... for items, n in (
|
|
|
|
... ([10, 20, 30, 40], 2), ([1, 2, 3], 1), ([1, 2, 3], 3), ([42], 1),
|
|
|
|
... ([6, 7, 8, 9], 4), ([10, 20, 30, 40, 50], 1), ([1, 2, 3, 4], 2),
|
|
|
|
... ([1, 'apple', 3.14], 2), (['single'], 0), ([], 9)))
|
|
|
|
True
|
2020-05-22 06:10:11 +00:00
|
|
|
"""
|
2023-10-15 19:56:02 +00:00
|
|
|
r = len(elements)
|
|
|
|
if n > r:
|
|
|
|
return []
|
|
|
|
|
|
|
|
dp: list[list[tuple]] = [[] for _ in range(r + 1)]
|
2019-10-05 05:14:13 +00:00
|
|
|
|
2023-10-15 19:56:02 +00:00
|
|
|
dp[0].append(())
|
2019-10-05 05:14:13 +00:00
|
|
|
|
2023-10-15 19:56:02 +00:00
|
|
|
for i in range(1, r + 1):
|
|
|
|
for j in range(i, 0, -1):
|
|
|
|
for prev_combination in dp[j - 1]:
|
2024-04-02 19:18:47 +00:00
|
|
|
dp[j].append((*prev_combination, elements[i - 1]))
|
2023-10-15 19:56:02 +00:00
|
|
|
|
|
|
|
try:
|
|
|
|
return sorted(dp[n])
|
|
|
|
except TypeError:
|
|
|
|
return dp[n]
|
2019-10-05 05:14:13 +00:00
|
|
|
|
|
|
|
|
2022-10-22 11:33:51 +00:00
|
|
|
if __name__ == "__main__":
|
2023-10-15 19:56:02 +00:00
|
|
|
from doctest import testmod
|
|
|
|
|
|
|
|
testmod()
|
|
|
|
print(f"{subset_combinations(elements=[10, 20, 30, 40], n=2) = }")
|