Post image
Itertools. Part2

I’m glad you like the previous article where I showed you some CheckiO solutions using combinatoric generators from itertools.

I have decided not to wait for too long and show you some other functions and solutions from CheckiO. I choose most popular based on statistic I have showed before.

chain - iterates from all all passed iterables

In: list(itertools.chain('ABC', 'DEF'))
Out: ['A', 'B', 'C', 'D', 'E', 'F']

In: list(itertools.chain('ABC', 'DEF', [1,2,3]))
Out: ['A', 'B', 'C', 'D', 'E', 'F', 1, 2, 3]

Solutions: gyahun_dash's solution for How to find friends, And again, gyahun_dash's and his solution for Cipher Crossword

cycle - iterate through iterator over and over again

cycle('ABCD') --> A B C D A B C D A B C D …

Solutions: DiZ’s solution for Vigenere Cipher is also using accumulate function with cycle.

accumulate - accomulates iterate values while iterate. (starting with Python3)

In: list(itertools.accumulate([1,2,3,4,5]))
Out: [1, 3, 6, 10, 15]

In: list(itertools.accumulate('ABC'))
Out: ['A', 'AB', 'ABC']

By default - summing, but it can be changed

In: list(itertools.accumulate([1,2,3,4,5], lambda a, b: a*b))
Out: [1, 2, 6, 24, 120]

Solutions: I can’t avoid veky’s creative solutions for Vigenere Cipher

repeat - works the same way as cycle, but this time it repeats the same element endlessly. Repeat has an additional arguments (times), with this argument it will be repeated limited amount of times

repeat(8) --> 8 8 8 8 8 ...

repeat(10, 3) --> 10 10 10

Solutions: eiichi solution of Speech Module might be an example of using repeat.

Now let’s check some really interesting function, such as groupby that makes an iterator that returns consecutive keys and groups from the iterable.

In: [[a, list(b)] for a,b in itertools.groupby('AAAABBBBCCCSSS')]
Out: 
[['A', ['A', 'A', 'A', 'A']],
 ['B', ['B', 'B', 'B', 'B']],
 ['C', ['C', 'C', 'C']],
 ['S', ['S', 'S', 'S']]]

In: [[a, list(b)] for a,b in itertools.groupby('AAAABBBBAAAA')]
Out: 
[['A', ['A', 'A', 'A', 'A']],
 ['B', ['B', 'B', 'B', 'B']],
 ['A', ['A', 'A', 'A', 'A']]]

groupby has an additional parameter key, that allows you to define grouping rules

In [9]: [[a, list(b)] for a,b in itertools.groupby('AAAABBBaaabbBCaCCSSssSss', key=lambda a: a.isupper())]
Out[9]: 
[[True, ['A', 'A', 'A', 'A', 'B', 'B', 'B']],
 [False, ['a', 'a', 'a', 'b', 'b']],
 [True, ['B', 'C']],
 [False, ['a']],
 [True, ['C', 'C', 'S', 'S']],
 [False, ['s', 's']],
 [True, ['S']],
 [False, ['s', 's']]]

Solutions: nickie's solution for Find Sequence, almoust oneliner from Adam.Michalak’s solution of Verify anagrams, makoto_yamagata solution of Spaceship landing strip

zip_longest makes an iterator that aggregates elements from each of the iterables. If the iterables are of uneven length, missing values are filled-in with fillvalue.

In: list(itertools.zip_longest([1,2,3,5], [4,5,6]))
Out: [(1, 4), (2, 5), (3, 6), (5, None)]

In: list(itertools.zip_longest([1,2,3,5], [4,5,6], fillvalue=0))
Out: [(1, 4), (2, 5), (3, 6), (5, 0)]

Solutions: Absolute champion that use this function is PositronicLlama’s solution of Funny addition, bryukh’s very well commented solution for The Hidden Word

starmap is basically map version2. Roughly equivalent to:

def starmap(function, iterable):
    for args in iterable:
        yield function(*args)

In: list(itertools.starmap(pow, [(2,5), (3,2), (10,3)]))
Out: [32, 9, 1000]

Solutions: Creative solution of Roman Numerals from gyahun_dash

takewhile makes an iterator that returns elements from the iterable as long as the predicate is true.

In: list(itertools.takewhile(lambda x: x<5, [1,4,6,4,1]))
Out: [1, 4]

Solutions: bryukh’s solution for Spaceship landing strip, Feed Pigeons by ale1ster, gyahun_dash’s solution for IP Network: Route Summarization

Even though on CheckiO you can find a lot of creative solutions - Python documentation has some interesting recipes as well.

PS: Special request from veky :)

compress (New in version 3.1) makes an iterator that filters elements from data returning only those that have a corresponding element in selectors that evaluates to True and roughly equivalent to:

def compress(data, selectors):
    # compress('ABCDEF', [1,0,1,0,1,1]) --> A C E F
    return (d for d, s in zip(data, selectors) if s)

In: list(itertools.compress('ABC', [1,0,1]))
Out: ['A', 'C']

Solutions: Pohmelie’s solution for Loading Cargo, veghadam1991’s solution for Stair steps, And.. suddenly solutions for Fizz Buzz by nilp0inter

Created: Feb. 2, 2017, 2:23 p.m.
9
40
User avatar
oduvan