Conjoin function made in functional style

后端 未结 2 382
情深已故
情深已故 2021-01-13 01:46

Recently, reading Python \"Functional Programming HOWTO\", I came across a mentioned there test_generators.py standard module, where I found the following gener

相关标签:
2条回答
  • 2021-01-13 02:25

    simple_conjoin uses the same basic building blocks -- loops, conditions, and yield -- as the building blocks of the itertools recipes. It also treats functions as data, a hallmark of functional programming.

    Of course this is most useful when the iterators have side-effects, so that which values can be generated at each slot depend on the values iterated at previous slots.

    This, however, is contrary to the way functional programming works. In functional programming, each function takes input and produces output, and reacts with the rest of the program in no other way.

    In simple_conjoin, the functions take no input, and have side effects. This is central to it's use.

    So while you can certainly write it in functional style, it won't be useful in simple translation.

    You'd need to figure out a way to write it so it operated without side effects before you could produce a truly "functional" implementation.

    Note: @recursive's answer is good, but if range3 had side effects it wouldn't be truly functional.

    0 讨论(0)
  • 2021-01-13 02:36

    This seems to work, and it's still lazy:

    def conjoin(gs):
        return [()] if not gs else (
            (val,) + suffix for val in gs[0]() for suffix in conjoin(gs[1:])
        )
    
    def range3():
        return range(3)
    
    print list(conjoin([range3, range3]))
    

    Output:

    [(0, 0), (0, 1), (0, 2), (1, 0), (1, 1), (1, 2), (2, 0), (2, 1), (2, 2)]
    

    Example usage to show mutable state:

    x = ""
    def mutablerange():
        global x
        x += "x"
        return [x + str(i) for i in range(3)]
    
    print list(conjoin([range3, mutablerange]))
    

    Output: (watch the increasing number of 'x's)

    [(0, 'x0'), (0, 'x1'), (0, 'x2'), (1, 'xx0'), (1, 'xx1'), (1, 'xx2'), (2, 'xxx0'), (2, 'xxx1'), (2, 'xxx2')]
    

    And if we use itertools.product:

    x = ""
    print list(itertools.product(range3(), mutablerange()))
    

    the result is the following:

    [(0, 'x0'), (0, 'x1'), (0, 'x2'), (1, 'x0'), (1, 'x1'), (1, 'x2'), (2, 'x0'), (2, 'x1'), (2, 'x2')]
    

    So, one clearly see, that itertools.product caches the values returned by the iterator.

    0 讨论(0)
提交回复
热议问题