Long time ago, various people, including me, were trying to make permutations and combinations with nested loops and various other “heavy” structures. Anyway, in Python there is a library, which does this for us and sets the end of the useless nested loops or nested loops with recursion. The library is quite understandable and is called itertools.
In this article, I will try to present a few “magic features” of itertools, as my own go-to place, if I have to do it again.
All possible combinations there…
Given the list(1,2,3,4), you may like to see all lists of n elements with the elements of this list. If n is 3, then this is the result of your expectations:
1 2 3 4 |
[(1, 1, 1), (1, 1, 2), (1, 1, 3), (1, 1, 4), (1, 2, 1), (1, 2, 2), (1, 2, 3), (1, 2, 4), (1, 3, 1), (1, 3, 2), (1, 3, 3), (1, 3, 4), (1, 4, 1), (1, 4, 2), (1, 4, 3), (1, 4, 4), (2, 1, 1), (2, 1, 2), (2, 1, 3), (2, 1, 4), (2, 2, 1), (2, 2, 2), (2, 2, 3), (2, 2, 4), (2, 3, 1), (2, 3, 2), (2, 3, 3), (2, 3, 4), (2, 4, 1), (2, 4, 2), (2, 4, 3), (2, 4, 4), (3, 1, 1), (3, 1, 2), (3, 1, 3), (3, 1, 4), (3, 2, 1), (3, 2, 2), (3, 2, 3), (3, 2, 4), (3, 3, 1), (3, 3, 2), (3, 3, 3), (3, 3, 4), (3, 4, 1), (3, 4, 2), (3, 4, 3), (3, 4, 4), (4, 1, 1), (4, 1, 2), (4, 1, 3), (4, 1, 4), (4, 2, 1), (4, 2, 2), (4, 2, 3), (4, 2, 4), (4, 3, 1), (4, 3, 2), (4, 3, 3), (4, 3, 4), (4, 4, 1), (4, 4, 2), (4, 4, 3), (4, 4, 4)] |
The trick is that n can be bigger than the list, and then the results become quite a lot… E.g. with n = 7 we get 16384 different lists, starting from (1,1,1,1,1,1,1) and finishing with (4,4,4,4,4,4,4). The good thing is that you do not need to take care of it and you would not get crazy writing nested loops with recursion. It is that easy with Python:
1 2 3 4 5 |
import itertools import json my_product = itertools.product((1, 2, 3, 4), repeat=3) print (json.dumps(list(my_product), indent = 2, sort_keys = True)) |
Only combinations with non-repeatable (unique) elements, where order matters
If we want to have all the 3 digit lists from (1, 2, 3, 4), where the elements are not to be repeated, it is quite ok as well – itertools.permutations((1, 2, 3, 4), 3) does the magic for us:
1 2 |
my_permutation = itertools.permutations((1, 2, 3, 4), 3) print (list(my_permutation)) |
resulting in:
1 2 |
[(1, 2, 3), (1, 2, 4), (1, 3, 2), (1, 3, 4), (1, 4, 2), (1, 4, 3), (2, 1, 3), (2, 1, 4), (2, 3, 1), (2, 3, 4), (2, 4, 1), (2, 4, 3), (3, 1, 2), (3, 1, 4), (3, 2, 1), (3, 2, 4), (3, 4, 1), (3, 4, 2), (4, 1, 2), (4, 1, 3), (4, 2, 1), (4, 2, 3), (4, 3, 1), (4, 3, 2)] |
Combinations where order does not matter – like in the lottery
The third thing to be presented here is a set of combinations, in which the order does not matter. It is like in the lottery – you “only” need to guess 6 numbers correctly and their order is not relevant. In our case, all the combinations of 3 elements from a list with 4 numbers are the following:
1 |
[(1, 2, 3), (1, 2, 4), (1, 3, 4), (2, 3, 4)] |
Yup, just 4. Got with the following code:
1 2 |
my_combination = itertools.combinations((1, 2, 3, 4), 3) print(list(my_combination)) |
If you are interested to see the total number of combinations from 6/49 lottery and you cannot calculate the following factorials:
nor you are interesting in searching in google for the answer, then you may run the code below:
1 2 3 4 |
lottery_numbers = list(range(1,50)) print(lottery_numbers) lottery_combinations = itertools.combinations(lottery_numbers, 6) print(len(list(lottery_combinations))) |
It returns exactly 13983816, which is the count of the combinations of the lotto 6/49. Almost 14 million!
🙂