The itertools
is a built-in module in Python 3 and is helpful for us to use a more pythonic way to manipulate the iterables (e.g., list, set, tuple, etc.).
Another advantage of itertools
is it also returns the generator type instances that have benefits of lazy evaluation from all of its methods.
In this section, we will discuss the second part of itertools
. The methods in this part all return a generator that generates finite items corresponding to your input.
accumulate
works like reduce, but accumulate
keep tracking the process of calculation and return in a generator.
This table takes Python documentation #accumulate as reference.
Arguments | Results | Example |
---|---|---|
p (iterable), [func=operator.add] |
p[0], p[0]+p[1], p[0]+p[1]+p[2], ... | accumulate([1, 2, 3]) = 1, 3, 6 |
import operator
from itertools import accumulate
gen = accumulate([1, 2, 3, 4])
list(gen) # [1, 3, 6, 10]
gen = accumulate([1, 2, 3, 4], func=operator.mul)
list(gen) # [1, 2, 6, 24]
This table takes Python documentation #chain as reference.
Arguments | Results | Example |
---|---|---|
p, q (iterables) |
p[0], p[1], ... p[-1], q[0], q[1], ... q[-1] | chain("AB", "CD") = A, B, C, D |
from itertools import chain
gen = chain([1, 2], [3, 4])
list(gen) # [1, 2, 3, 4]
gen = chain("AB", "CD")
list(gen) # [A, B, C, D]
This table takes Python documentation #compress as reference.
Arguments | Results | Example |
---|---|---|
p, selector (iterables) |
(p[0] if s[0]), (p[1] if s[1]), ... | compress("AB", [0, 1]) = B |
from itertools import compress
gen = compress([1, 2, 3], [1, 0, 1])
gen = compress([1, 2, 3], [True, False, True]) # same
list(gen) # [1, 3]
filterfalse
is basically the opposite version of filter.
This table takes Python documentation #filterfalse as reference.
Arguments | Results | Example |
---|---|---|
pred, p (iterable) |
p[i] != pred | filterfalse(lambda x: x%2==0, [1, 2, 3]) = 1, 3 |
from itertools import filterfalse
gen = filterfalse(lambda x: x%2 == 0, [1, 2, 3])
list(gen) # [1, 3]
groupby
will separate the iterable (e.g., list
, dict
, tuple
) into key-generator pairs
based on the keyfunc
(default is labmda x: x
).
It is important to note that it is not like the "groupby" in SQL where the data is sorted before the group; If you want to perfectly separate the iterable, you should sort it before groupby
.
This table takes Python documentation #groupby as reference.
Arguments | Results | Tuples | Example |
---|---|---|---|
p (iterable)[key=lambda x: x] |
tuple1, tuple2, tuple3, ... | (key, generator) | groupby("AABBCCCAA") = [(A, AA), (B, BB), (C, CCC), (A, AA)] |
from itertools import groupby
gen = groupby("AABBCCCAA") # default func = lambda x: x
for k, g in gen:
print(k, list(g))
# A [A, A]
# B [B, B]
# C [C, C, C]
# A [A, A]
gen = groupby([1, 2, 3, 4], lambda x: x // 3)
for k, g in gen:
print(k, list(g))
# 0 [1, 2]
# 1 [3, 4]
gen = groupby([("A", 100), ("B", 200), ("C", 600)], lambda x: x[1] > 500)
for k, g in gen:
print(k, list(g))
# False [(A, 100), (B, 200)]
# True [(C, 600)]
The function of islice
is the same as the slice function
in the list (e.g., list[0:4:2]), the only difference is it returns a generator.
This table takes Python documentation #islice as reference.
Arguments | Results | Example |
---|---|---|
p (iterable), [start=None], stop, [step=None] |
p[start:stop:step] but in generator | islice("ABCD", 0, None, 2) = A, C |
gen = islice([1, 2, 3], 2) # equals to A[:2]
list(gen) # [1, 2]
gen = islice("ABCD", 2, 4) # equals to A[2:4]
list(gen) # [C, D]
gen = islice("ABCD", 0, None, 2) # equals to A[::2]
list(gen) # [A, C]
The function of starmap
is almost the same as map, except that starmap
treats elements directly as func arguments in the same order.
This table takes Python documentation #starmap as reference.
Arguments | Results | Example |
---|---|---|
func, p (iterable) |
func(*p[0]), ..., func(p[-1]) | starmap(pow, [(2, 3), (2, 4)]) = 8, 16 |
from itertools import starmap
# with only one argument
gen = starmap(lambda x: x.lower(), "ABCD")
list(gen) # [a, b, c, d]
# with 2 arguments
gen = starmap(lambda x, y: x + y, [(1, 2), (3, 4)])
list(gen) # [3, 7]
# with different size of arugments
gen = starmap(lambda *keys: sum(keys) / len(keys), [[3, 8, 3], [4, 2]])
list(gen) # [4.6666667, 3.0]
This table takes Python documentation #takewhile as reference.
Arguments | Results | Example |
---|---|---|
pred, p (iterable) |
p[0], p[1], ... until pred fails | takewhile(lambda x: x<3, [1, 2, 3, 2]) = 1, 2 |
from itertools import takewhile
gen = takewhile(lambda x: x < 2, [1, 2, 3, 2, 1])
list(gen) # [1]
gen = takewhile(lambda x: x.isupper(), "ABCdefgHIJ")
list(gen) # [A, B, C]
This table takes Python documentation #dropwhile as reference.
Arguments | Results | Example |
---|---|---|
pred, p (iterable) |
starting when pred fails, p[n], p[n+1], ... | dropwhile(lambda x: x<3, [1, 2, 3, 2]) = 3, 2 |
gen = dropwhile(lambda x: x < 2, [1, 2, 3, 2, 1])
list(gen) # [2, 3, 2, 1]
gen = dropwhile(lambda x: x.isupper(), "ABCdefgHIJ")
list(gen) # [d, e, f, g, H, I, J]
The function of zip_longest
is almost the same as zip, except that zip_longest
will fill the missing value with fillvalue
(default is None
) if the iterables are with uneven length.
This table takes Python documentation #zip_longest as reference.
Arguments | Results | Example |
---|---|---|
p, q (iterables), [fillvalue=None] |
(p[0], q[0]), (p[1], q[1]), ... | zip_longest("ABC", [1, 2], fillvalue=-1) = (A, 1), (B, 2), (C, -1) |
from itertools import zip_longest
gen = zip_longest("ABC", ("X", "Y"))
list(gen) # [('A', 'X'), ('B', 'Y'), ('C', None)]
gen = zip_longest("ABC", [1, 2], fillvalue=-1)
list(gen) # [('A', 1), ('B', 2), ('C', -1)]
Article | Link |
---|---|
Python documentation - itertools | https://docs.python.org/3/library/itertools.html |
Python Tutorial: Itertools Module - Iterator Functions for Efficient Looping | https://www.youtube.com/watch?v=Qu3dThVy6KQ |
Python 好用模組介紹 - itertools & more-itertools | https://myapollo.com.tw/zh-tw/python-itertools-more-itertools/ |
Python標準庫之itertools庫的使用方法 | https://codertw.com/%E7%A8%8B%E5%BC%8F%E8%AA%9E%E8%A8%80/364249/ |
python groupby 筆記 | http://ot-note.logdown.com/posts/1379449 |