In connection with functional programming (see also wikipedia) we start with a few words about how do we decompose a program, that is about the programming paradigms. There are some main types of them, but several programming languages cannot be strictly classified into one paradigm, but rather include features from several paradigms, like Python.
In the functional paradigm any function $f(x)$ results the same value for the same $x$.
This is not the case if you have variables and change them while the program is running. Look at the following function (result depends on $x$).
def f(y):
return x + y
x = 1
print(f(5))
x = -5
print(f(5))
If you avoid those, then the python functions will be actual mathematical functions), otherwise those are sequence of commands. Hence the name functional programming.
Originally, Guido van Rossum wanted to drop out the main functional programming tools from the core of Python 3 into a modul, because python has list comprehension instead.
But after a strong lobby of functional programming fanatics Rossum kept the lambda
, map()
and filter()
in the core and moved reduce()
to the functools
package.
The theoretical and practical advantages to the functional style are
The first and most important functional programming element of Python is the iterator what we introduced before. Running through the elements of an iterable object and performing an operation on them or selecting some of them by a condition are the most important tasks in a functional programming approach. In Python one may use list comprehension or generator expression for these tasks. List comprehension gives back a list, generator expression an iterator. The only difference is that we put the former in []
and the latter in ()
.
Example: After reading the lines from a file into a list of strings, remove the white space characters from the beginning and end of each line, using the .strip()
method.
lines = [' line1 ', 'line2\t', ' ', ' line3 \n ', '\n']
stripped_iter = (line.strip() for line in lines)
for i in stripped_iter:
print(i, end="")
line1line2line3
[line.strip() for line in lines]
['line1', 'line2', '', 'line3', '']
[line.strip() for line in lines if line.strip() != ""]
['line1', 'line2', 'line3']
The general form of the generator expression:
(expression for expr1 in seq1
if condition1
for expr2 in seq2
if condition2
...
for exprN in seqN
if conditionN )
The list comprehension is the same except that []
is used instead of ()
.
These are equivalent to the next code. Mind the position of the main expression
:
for expr1 in seq1:
if not (condition1):
continue
for expr2 in seq2:
if not (condition2):
continue
...
for exprN in seqN:
if not (conditionN):
continue
expression
[(x, y, x * y) for x in (1, 2, 3) for y in [1, 2, 3, 4] if x < y]
[(1, 2, 2), (1, 3, 3), (1, 4, 4), (2, 3, 6), (2, 4, 8), (3, 4, 12)]
l = []
for x in (1, 2, 3):
for y in [1, 2, 3, 4]:
if x < y:
l.append((x, y, x*y))
l
[(1, 2, 2), (1, 3, 3), (1, 4, 4), (2, 3, 6), (2, 4, 8), (3, 4, 12)]
Exercise: Print out the leap years between 1890 and 1914!
[year for year in range(1890, 1915)];
[year for year in range(1890, 1915) if year%4 == 0]
[year for year in range(1890, 1915)
if (year%4 == 0 and year%100 != 0) or year%400 == 0]
Some simple functions can be written shorter with lambda. Traditional function, the form of it is
lambda arguments: expression
def square(x):
return x * x
square(5)
25
You may define this function wit lambda
:
square2 = lambda x: x*x
square2(6)
36
(lambda x: x**2)(6)
36
A lambda function can have more arguments:
expr = lambda a, b, c: a + b*c
expr(2, 4, 4)
18
The map
applies a given function for every element of an iterable object and returns an iterator. Describing with a math formula:
$$\mathop{\mathrm{map}}:(A\mapsto B)\times A^n\mapsto B^n\ \ (n\in \mathbb{N})$$
l = [1, 5, 8]
list(map(square, l))
[1, 25, 64]
The same with lambda (and list comprehension):
list(map(lambda x: x**2, range(5)))
[0, 1, 4, 9, 16]
[x**2 for x in range(5)]
[0, 1, 4, 9, 16]
You can use a multi-variate function as the first argument. If $f$ has more variable, then you have to provide more iterables.
$$\mathop{\mathrm{map}}: (A \mapsto X)\times A^n \mapsto X^n$$$$\mathop{\mathrm{map}}: (A\times B \mapsto X) \times A^n\times B^n \mapsto X^n$$$$\mathop{\mathrm{map}}: (A\times B \times C\mapsto X)\times A^n\times B^n\times C^n \mapsto X^n$$$$\vdots$$map(pow, [0.5, 1, 1.5], [-1, 2, 3])
<map at 0x7ffbec394be0>
list(map(pow, [0.5, 1, 1.5], [-1, 2, 3]))
[2.0, 1, 3.375]
list(map(pow, [0.5, 1], [-1, 2, 3]))
[2.0, 1]
We may use conditional expressions in lambda functions. Let us recall that the general form of a conditional expression is
expression if condition else expression_when_condition_false
x = -4
x if x>0 else -x
4
absolute = lambda x: x if x >= 0 else -x
print(absolute(5), absolute(-5))
5 5
list(map(lambda x: x if x>=0 else 0, range(-5,5)))
One of the advantage of using conditional expressions is that both if
and else
are necessary, which makes it harder to forget a case.
Bad example:
def ageroup(x):
if x < 8:
return "child"
elif x < 15:
return "adolescent"
elif x >=18:
return "adult"
print(ageroup(16))
def ageroup(x):
return "child" if x<8 else ("adolescent"
if x<15 else ("adult" if x>=18 else "INVALID"))
ageroup(16)
The filter
makes an iterator from an iterable object according to a condition given as a logical function.
The logical function should be $A\mapsto \{True, False\}$.
$$\mathrm{filter} : (A\mapsto \{True, False\})\times A^n\mapsto A^m \text{ ahol } m\leq n$$list(filter(lambda x: x % 2 == 0, range(10)))
[x for x in range(10) if x % 2 == 0]
list(filter(str.isupper, "aAbBcC"))
zip()
¶list(zip(['a', 'b', 'c'], (1, 2, 3)))
any()
és all()
¶According to the quantifiers the any()
and all()
built-ins look at the truth values of an iterable object, and any()
returns True
if any element in the iterable is true, and all()
returns True
if all of the elements are true.
any([1,2,0,1])
all([1,2,0,1])
import math
print([math.cos(i*math.pi) for i in range(3)])
any([math.cos(i*math.pi) == 1.0 for i in range(3)])
all([math.cos(i*math.pi) == 1.0 for i in range(3)])
The functools.reduce(*function*, *iterable*[, *initial*])
cumulatively performs a bi-variate function on all the iterable’s elements.
functools.reduce()
takes the first two elements a
and b
returned by the iterator and calculates func(a, b)
, then calculates func(func(a, b), c)
with the third element, and continues until the iterable is exhausted. If an initial value is given, then the calculation starts with func(initial, a)
. If only one element is offered by the iterator, then that will be the result. If no element is offered by the iteratot then a TypeError
exception is raised. So always use an initial value if this may happen.
from functools import reduce
addition = lambda a, b: a+b
reduce(addition, [1, 2, 3, 4])
reduce(addition, [4])
reduce(addition, [], 0)
division = lambda a, b: a/b
reduce(division, [2, 3, 4], 1)
1/24
The paradigm of accumulation can be done with this. Finding the extrema can be one application.
greater = lambda x, y: x if x > y else y
reduce(greater, [0, 1, 10, -10, 2], float("-inf"))
smaller = lambda x, y: x if x < y else y
reduce(smaller, [0, 1, 10, -10, 2], float("inf"))
def dot(a, b):
return sum(map(lambda x, y: x*y, a, b))
dot([0.5, 1, 1.5], [-1, 1, -1])
In general, expressions of the form $f(g(x_1, y_1), g(x_2, y_2), \ldots g(x_n, y_n))$ can be considered a generalized inner product.
It is easy to implement functionally:
def inner(a, b, g=lambda x, y: x*y, f=sum):
return f(map(g, a, b))
inner([0.5, 1, 1.5], [-1, 1, -1])
inner("abc", [1,3,2], f=list)
Or with a bi-variate, asssociative $g$ function via a reduce:
def inner(a, b, g=lambda x, y: x*y, f=lambda a, b: a+b, i=0):
return reduce(f, map(g, a, b), i)
inner([0.5, 1, 1.5], [-1, 1, -1])
inner("abc", [1,3,2], i='')
inner([1,11,4], [3,2,6], g=min, f=max, i=float("-Inf"))
inner([1,11,4], [3,2,6], g=max, f=min, i=float("Inf"))
Or do something similar with the 'or's of 'and's or with the 'and's of 'or's. For this we need a function version of the operations or
and and
. We may construct these functions or we may use the operations
modul:
or_ = lambda a, b: a or b
and_ = lambda a, b: a and b
from operator import or_, and_
inner([0,0,1,1], [0,1,0,1], g=or_, f=and_, i=1)
inner([0,0,1,1], [0,1,0,1], g=and_, f=or_, i=0)
The eight queens puzzle is the problem of placing eight chess queens on an $8\times8$ chessboard so that no two queens threaten each other. A solution requires that no two queens share the same row, column, or diagonal. How many solutions are there? List them!
Two queens in the line $i$ and $j$ ($i\neq j$) does not threaten each other if $\mathrm{abs}(i - j)\neq\mathrm{abs}(x[i] - x[j])$. Listing the permutations we import the function permutations
from the itertools
module. This iterates through the permutations of the elements.
from itertools import permutations
s = 0
for x in permutations(list(range(8))):
if all([i == j or abs(i - j) != abs(x[i] - x[j])
for i in range(8) for j in range(8)]):
s += 1
print(s)
92
for x in permutations(list(range(8))):
if all([i == j or abs(i - j) != abs(x[i] - x[j])
for i in range(8) for j in range(8)]):
print(x)
(0, 4, 7, 5, 2, 6, 1, 3) (0, 5, 7, 2, 6, 3, 1, 4) (0, 6, 3, 5, 7, 1, 4, 2) (0, 6, 4, 7, 1, 3, 5, 2) (1, 3, 5, 7, 2, 0, 6, 4) (1, 4, 6, 0, 2, 7, 5, 3) (1, 4, 6, 3, 0, 7, 5, 2) (1, 5, 0, 6, 3, 7, 2, 4) (1, 5, 7, 2, 0, 3, 6, 4) (1, 6, 2, 5, 7, 4, 0, 3) (1, 6, 4, 7, 0, 3, 5, 2) (1, 7, 5, 0, 2, 4, 6, 3) (2, 0, 6, 4, 7, 1, 3, 5) (2, 4, 1, 7, 0, 6, 3, 5) (2, 4, 1, 7, 5, 3, 6, 0) (2, 4, 6, 0, 3, 1, 7, 5) (2, 4, 7, 3, 0, 6, 1, 5) (2, 5, 1, 4, 7, 0, 6, 3) (2, 5, 1, 6, 0, 3, 7, 4) (2, 5, 1, 6, 4, 0, 7, 3) (2, 5, 3, 0, 7, 4, 6, 1) (2, 5, 3, 1, 7, 4, 6, 0) (2, 5, 7, 0, 3, 6, 4, 1) (2, 5, 7, 0, 4, 6, 1, 3) (2, 5, 7, 1, 3, 0, 6, 4) (2, 6, 1, 7, 4, 0, 3, 5) (2, 6, 1, 7, 5, 3, 0, 4) (2, 7, 3, 6, 0, 5, 1, 4) (3, 0, 4, 7, 1, 6, 2, 5) (3, 0, 4, 7, 5, 2, 6, 1) (3, 1, 4, 7, 5, 0, 2, 6) (3, 1, 6, 2, 5, 7, 0, 4) (3, 1, 6, 2, 5, 7, 4, 0) (3, 1, 6, 4, 0, 7, 5, 2) (3, 1, 7, 4, 6, 0, 2, 5) (3, 1, 7, 5, 0, 2, 4, 6) (3, 5, 0, 4, 1, 7, 2, 6) (3, 5, 7, 1, 6, 0, 2, 4) (3, 5, 7, 2, 0, 6, 4, 1) (3, 6, 0, 7, 4, 1, 5, 2) (3, 6, 2, 7, 1, 4, 0, 5) (3, 6, 4, 1, 5, 0, 2, 7) (3, 6, 4, 2, 0, 5, 7, 1) (3, 7, 0, 2, 5, 1, 6, 4) (3, 7, 0, 4, 6, 1, 5, 2) (3, 7, 4, 2, 0, 6, 1, 5) (4, 0, 3, 5, 7, 1, 6, 2) (4, 0, 7, 3, 1, 6, 2, 5) (4, 0, 7, 5, 2, 6, 1, 3) (4, 1, 3, 5, 7, 2, 0, 6) (4, 1, 3, 6, 2, 7, 5, 0) (4, 1, 5, 0, 6, 3, 7, 2) (4, 1, 7, 0, 3, 6, 2, 5) (4, 2, 0, 5, 7, 1, 3, 6) (4, 2, 0, 6, 1, 7, 5, 3) (4, 2, 7, 3, 6, 0, 5, 1) (4, 6, 0, 2, 7, 5, 3, 1) (4, 6, 0, 3, 1, 7, 5, 2) (4, 6, 1, 3, 7, 0, 2, 5) (4, 6, 1, 5, 2, 0, 3, 7) (4, 6, 1, 5, 2, 0, 7, 3) (4, 6, 3, 0, 2, 7, 5, 1) (4, 7, 3, 0, 2, 5, 1, 6) (4, 7, 3, 0, 6, 1, 5, 2) (5, 0, 4, 1, 7, 2, 6, 3) (5, 1, 6, 0, 2, 4, 7, 3) (5, 1, 6, 0, 3, 7, 4, 2) (5, 2, 0, 6, 4, 7, 1, 3) (5, 2, 0, 7, 3, 1, 6, 4) (5, 2, 0, 7, 4, 1, 3, 6) (5, 2, 4, 6, 0, 3, 1, 7) (5, 2, 4, 7, 0, 3, 1, 6) (5, 2, 6, 1, 3, 7, 0, 4) (5, 2, 6, 1, 7, 4, 0, 3) (5, 2, 6, 3, 0, 7, 1, 4) (5, 3, 0, 4, 7, 1, 6, 2) (5, 3, 1, 7, 4, 6, 0, 2) (5, 3, 6, 0, 2, 4, 1, 7) (5, 3, 6, 0, 7, 1, 4, 2) (5, 7, 1, 3, 0, 6, 4, 2) (6, 0, 2, 7, 5, 3, 1, 4) (6, 1, 3, 0, 7, 4, 2, 5) (6, 1, 5, 2, 0, 3, 7, 4) (6, 2, 0, 5, 7, 4, 1, 3) (6, 2, 7, 1, 4, 0, 5, 3) (6, 3, 1, 4, 7, 0, 2, 5) (6, 3, 1, 7, 5, 0, 2, 4) (6, 4, 2, 0, 5, 7, 1, 3) (7, 1, 3, 0, 6, 4, 2, 5) (7, 1, 4, 2, 0, 6, 3, 5) (7, 2, 0, 5, 1, 4, 6, 3) (7, 3, 0, 2, 5, 1, 6, 4)