Python combinations from multiple lists
Is there any pythonic method for creating combinations between multiple lists? (similar to Cartesian product, but more complicated)
Example:
a = [1, 2, 3]
b = [4, 5, 6]
c = [7, 8, 9]
# ...
# there are more than 3 lists
Expected Result:
1. [(1, 4, 7), (2, 5, 8), (3, 6, 9)]
2. [(1, 4, 8), (2, 5, 7), (3, 6, 9)]
3. [(1, 4, 9), (2, 5, 7), (3, 6, 8)]
4. [(1, 5, 7), (2, 4, 8), (3, 6, 9)]
5. ...
Update:
Thanks for the quick reply ~ !!
To clarify the question:
As a result, all repeated combinations of the Cartesian product of the list a, b, c.
This can be done with another ugly method:
1) Form the entire list of the Cartesian product
from itertools import product, combinations, chain
t = list(product(a, b, c))
2) Using combinations to generate all possible results
p = list(combinations(t, 3))
3) Filter out duplicate conditions
cnt = len(list(chain(a, b, c)))
f = [x for x in p if len(set(chain(*x))) == cnt]
Update2:
The expected output produced by the ugly method is:
((1, 4, 7), (2, 5, 8), (3, 6, 9)) ((1, 4, 7), (2, 5, 9), (3, 6, 8)) ((1, 4, 7), (2, 6, 8), (3, 5, 9)) ((1, 4, 7), (2, 6, 9), (3, 5, 8)) ((1, 4, 8), (2, 5, 7), (3, 6, 9)) ((1, 4, 8), (2, 5, 9), (3, 6, 7)) ((1, 4, 8), (2, 6, 7), (3, 5, 9)) ((1, 4, 8), (2, 6, 9), (3, 5, 7)) ((1, 4, 9), (2, 5, 7), (3, 6, 8)) ((1, 4, 9), (2, 5, 8), (3, 6, 7)) ((1, 4, 9), (2, 6, 7), (3, 5, 8)) ((1, 4, 9), (2, 6, 8), (3, 5, 7)) ((1, 5, 7), (2, 4, 8), (3, 6, 9)) ((1, 5, 7), (2, 4, 9), (3, 6, 8)) ((1, 5, 7), (2, 6, 8), (3, 4, 9)) ((1, 5, 7), (2, 6, 9), (3, 4, 8)) ((1, 5, 8), (2, 4, 7), (3, 6, 9)) ((1, 5, 8), (2, 4, 9), (3, 6, 7)) ((1, 5, 8), (2, 6, 7), (3, 4, 9)) ((1, 5, 8), (2, 6, 9), (3, 4, 7)) ((1, 5, 9), (2, 4, 7), (3, 6, 8)) ((1, 5, 9), (2, 4, 8), (3, 6, 7)) ((1, 5, 9), (2, 6, 7), (3, 4, 8)) ((1, 5, 9), (2, 6, 8), (3, 4, 7)) ((1, 6, 7), (2, 4, 8), (3, 5, 9)) ((1, 6, 7), (2, 4, 9), (3, 5, 8)) ((1, 6, 7), (2, 5, 8), (3, 4, 9)) ((1, 6, 7), (2, 5, 9), (3, 4, 8)) ((1, 6, 8), (2, 4, 7), (3, 5, 9)) ((1, 6, 8), (2, 4, 9), (3, 5, 7)) ((1, 6, 8), (2, 5, 7), (3, 4, 9)) ((1, 6, 8), (2, 5, 9), (3, 4, 7)) ((1, 6, 9), (2, 4, 7), (3, 5, 8)) ((1, 6, 9), (2, 4, 8), (3, 5, 7)) ((1, 6, 9), (2, 5, 7), (3, 4, 8)) ((1, 6, 9), (2, 5, 8), (3, 4, 7))
source to share
What you want is not combinations, but really permutations. 3 elements have 6 permutations, the Cartesian product of 2 sets of permutations has 36. PM 2Ring originally assumed you wanted all three of them to rearrange, since your question didn't say otherwise. If the code in the question produces the desired output, it means you want to move b
and c
, but not a
. I first wrote some code that calculated the permutations for all a
, b
and c
. However, since there is a
no need to rearrange, we'll just end it with a list. This brings us closer to the desired result:
import itertools as it a = [1, 2, 3] b = [4, 5, 6] c = [7, 8, 9] for i in it.product([tuple(a)], it.permutations(b), it.permutations(c)): print(i)
Output - 36 lines starting with
((1, 2, 3), (4, 5, 6), (7, 8, 9)) ((1, 2, 3), (4, 5, 6), (7, 9, 8)) ((1, 2, 3), (4, 5, 6), (8, 7, 9))
This is almost the same format you want, but with the indexes carried over, so it o[x][y]
will match o[y][x]
your desired output. We use zip
magic to transpose them. On the plus side, this function now works for any number of arguments:
import itertools as it
a = [1, 2, 3]
b = [4, 5, 6]
c = [7, 8, 9]
def funnyperms(first, *rest):
for i in it.product([first], *(it.permutations(j) for j in rest)):
yield tuple(zip(*i))
for i in funnyperms(a, b, c):
print(i)
Output signal
((1, 4, 7), (2, 5, 8), (3, 6, 9)) ((1, 4, 7), (2, 5, 9), (3, 6, 8)) ((1, 4, 8), (2, 5, 7), (3, 6, 9)) ((1, 4, 8), (2, 5, 9), (3, 6, 7)) ((1, 4, 9), (2, 5, 7), (3, 6, 8)) ((1, 4, 9), (2, 5, 8), (3, 6, 7)) ((1, 4, 7), (2, 6, 8), (3, 5, 9)) ((1, 4, 7), (2, 6, 9), (3, 5, 8)) ((1, 4, 8), (2, 6, 7), (3, 5, 9)) ((1, 4, 8), (2, 6, 9), (3, 5, 7)) ((1, 4, 9), (2, 6, 7), (3, 5, 8)) ((1, 4, 9), (2, 6, 8), (3, 5, 7)) ((1, 5, 7), (2, 4, 8), (3, 6, 9)) ((1, 5, 7), (2, 4, 9), (3, 6, 8)) ((1, 5, 8), (2, 4, 7), (3, 6, 9)) ((1, 5, 8), (2, 4, 9), (3, 6, 7)) ((1, 5, 9), (2, 4, 7), (3, 6, 8)) ((1, 5, 9), (2, 4, 8), (3, 6, 7)) ((1, 5, 7), (2, 6, 8), (3, 4, 9)) ((1, 5, 7), (2, 6, 9), (3, 4, 8)) ((1, 5, 8), (2, 6, 7), (3, 4, 9)) ((1, 5, 8), (2, 6, 9), (3, 4, 7)) ((1, 5, 9), (2, 6, 7), (3, 4, 8)) ((1, 5, 9), (2, 6, 8), (3, 4, 7)) ((1, 6, 7), (2, 4, 8), (3, 5, 9)) ((1, 6, 7), (2, 4, 9), (3, 5, 8)) ((1, 6, 8), (2, 4, 7), (3, 5, 9)) ((1, 6, 8), (2, 4, 9), (3, 5, 7)) ((1, 6, 9), (2, 4, 7), (3, 5, 8)) ((1, 6, 9), (2, 4, 8), (3, 5, 7)) ((1, 6, 7), (2, 5, 8), (3, 4, 9)) ((1, 6, 7), (2, 5, 9), (3, 4, 8)) ((1, 6, 8), (2, 5, 7), (3, 4, 9)) ((1, 6, 8), (2, 5, 9), (3, 4, 7)) ((1, 6, 9), (2, 5, 7), (3, 4, 8)) ((1, 6, 9), (2, 5, 8), (3, 4, 7))
Storing them in a set and comparing them with the values ββgenerated by your method proves that they have identical output:
print(set(funnyperms(a, b, c)) == set(f))
prints True
, QED
source to share
You can use the product from itertools for all combinations
>>> from itertools import product
>>> a = [1, 2, 3]
>>> b = [4, 5, 6]
>>> c = [7, 8, 9]
>>> A = [a,b,c]
>>> prod = list(product(*A))
>>> print(prod)
Expected Result:
[(1, 4, 7), (1, 4, 8), (1, 4, 9), (1, 5, 7), (1, 5, 8), (1, 5, 9), (1, 6, 7), (1, 6, 8), (1, 6, 9), (2, 4, 7), (2, 4, 8), (2, 4, 9), (2, 5, 7), (2, 5, 8), (2, 5, 9), (2, 6, 7), (2, 6, 8), (2, 6, 9), (3, 4, 7), (3, 4, 8), (3, 4, 9), (3, 5, 7), (3, 5, 8), (3, 5, 9), (3, 6, 7), (3, 6, 8), (3, 6, 9)]
source to share