1、前言
在《04.函数》一文中介绍了Python中的函数,以及函数的基础使用。函数是Python内建支持的一种封装,我们通过把大段代码拆成函数,通过一层一层的函数调用,就可以把复杂任务分解成简单的任务,这种分解可以称之为面向过程的程序设计。函数就是面向过程的程序设计的基本单元。而函数式编程(Functional Programming),是一种抽象程度很高的编程规范。
2、什么是函数式编程
函数式编程是一种编程范式,它将计算视为数学函数的评估,并避免改变状态和可变数据。它是一种抽象程度很高的编程范式,纯粹的函数式编程语言编写的函数没有变量,因此,任意一个函数,只要输入是确定的,输出就是确定的,这种纯函数我们称之为没有副作用。而允许使用变量的程序设计语言,由于函数内部的变量状态不确定,同样的输入,可能得到不同的输出,因此,这种函数是有副作用的。
函数式编程的一个特点就是,允许把函数本身作为参数传入另一个函数,还允许返回一个函数!在函数式编程中,函数是一等公民,允许其作为参数传递、赋值给变量,以及作为函数的返回值。
2.1、函数是一等公民
在Python中,函数是一等公民,这意味着函数可以被赋值给变量,作为参数传递给其他函数,以及作为函数的返回值。
代码语言:javascript复制def square(x):
return x ** 2
# 函数赋值给变量
f = square
# 函数作为参数传递
def apply_func(func, x):
return func(x)
result = apply_func(f, 5) # 结果为25
2.2、避免状态和可变数据
函数式编程鼓励使用不可变数据和避免副作用。这有助于减少程序的复杂性和提高代码的可维护性。
代码语言:javascript复制# 不可变数据
immutable_list = (1, 2, 3)
# 避免副作用
def add_to_list(element, lst):
return lst [element]
original_list = [1, 2, 3]
new_list = add_to_list(4, original_list) # new_list为[1, 2, 3, 4]
3、函数式编程的核心概念
从上面的例子可以看出,编写高阶函数,就是让函数的参数能够接收别的函数。把函数作为参数传入,这样的函数称为高阶函数,函数式编程就是指这种高度抽象的编程范式。
3.1、高阶函数
高阶函数是函数式编程的基石,它们可以接受一个或多个函数作为参数,并/或返回一个新的函数。Python中内置了很多高阶函数,如map/reduce、filter、sorted等。
- map函数是一个高阶函数,将函数应用于可迭代对象的每个元素。
if __name__ == '__main__':
numbers = [1, 2, 3, 4]
squared = map(lambda x: x ** 2, numbers) # 结果为[1, 4, 9, 16]
print(list(squared))
- reduce函数用于累积可迭代对象的元素,把一个函数作用在一个序列[x1, x2, x3, ...]上,这个函数必须接收两个参数,reduce把结果继续和序列的下一个元素做累积计算。
from functools import reduce
if __name__ == '__main__':
numbers = [1, 2, 3, 4]
total = reduce(lambda x, y: x y, numbers) # 结果为1 2 3 4=10
print(total)
- filter函数用于过滤可迭代对象的元素。
if __name__ == '__main__':
numbers = [1, 2, 3, 4, 5, 6]
even_numbers = filter(lambda x: x % 2 == 0, numbers) # 结果为[2, 4, 6]
print(list(even_numbers))
3.2、Lambda(匿名函数)
Lambda函数是一种简洁的函数定义方式,通常用于临时需要一个简单函数的情况。当我们在传入函数时,有些时候,不需要显式地定义函数,直接传入匿名函数更方便。在Python中,对匿名函数提供了有限支持。就像前面filter函数中的lambda x: x % 2 == 0,该匿名函数实际上是:
代码语言:javascript复制def func(x):
return x % 2 == 0
其中关键字lambda表示匿名函数,冒号前面的x表示函数参数。匿名函数有个限制,就是只能有一个表达式,不用写return,返回值就是该表达式的结果。用匿名函数有个好处,因为函数没有名字,不必担心函数名冲突。此外,匿名函数也是一个函数对象,也可以把匿名函数赋值给一个变量,再利用变量来调用该函数。如:
代码语言:javascript复制if __name__ == '__main__':
# 使用lambda定义匿名函数
multiply = lambda x, y: x * y
result = multiply(3, 4) # 结果为12
print(result)
3.3、递归 & 尾递归优化
函数式编程中,递归是一种强大的技术,可以通过函数调用自身来解决问题。
代码语言:javascript复制# 阶乘的递归实现
def factorial(n):
if n == 0 or n == 1:
return 1
else:
return n * factorial(n - 1)
在Python并不是尾递归优化的语言,但你可以使用一些技巧来模拟尾递归优化,如使用尾递归优化装饰器。
代码语言:javascript复制class TailRecursive:
def __init__(self, func):
self.func = func
self.args = None
self.kwargs = None
def __call__(self, *args, **kwargs):
self.args = args
self.kwargs = kwargs
while self.args is not None:
result = self.func(*self.args, **self.kwargs)
if callable(result):
result()
else:
self.args = None
self.kwargs = None
return result
# 使用尾递归优化装饰器
@TailRecursive
def factorial_tail_recursive(n, acc=1):
if n == 0:
return acc
else:
return factorial_tail_recursive(n - 1, n * acc)
result = factorial_tail_recursive(5) # 结果为120
3.4、functools模块
functools 是 Python 标准库中的一个模块,提供了一些与函数相关的高阶功能。它包含了一些用于函数操作的工具,其中一些特别有用于函数式编程。上面介绍reduce方法时,就已经有引入了该模块了。下面详细介绍几个常用方法,更多具体的可以详细查看API。
3.4.1、partial
该函数用于部分应用(partial application)一个函数,即固定函数的一些参数,返回一个新的函数。
代码语言:javascript复制from functools import partial
def power(base, exponent):
return base ** exponent
square = partial(power, exponent=2)
result = square(4) # 结果为16
3.4.2、reduce
该函数通常用于对可迭代对象的所有元素进行累积操作。
代码语言:javascript复制from functools import reduce
numbers = [1, 2, 3, 4]
product = reduce(lambda x, y: x * y, numbers) # 结果为24
3.4.3、lru_cache
该函数用于实现缓存机制,可以缓存函数的结果,以避免重复计算。
代码语言:javascript复制from functools import lru_cache
@lru_cache(maxsize=None)
def fibonacci(n):
if n <= 1:
return n
return fibonacci(n-1) fibonacci(n-2)
3.4.4、wraps
该函数用于在装饰器中正确处理被装饰函数的元信息,如文档字符串和函数名。
代码语言:javascript复制from functools import wraps
def my_decorator(f):
@wraps(f)
def wrapper(*args, **kwds):
print('Calling decorated function')
return f(*args, **kwds)
return wrapper
@my_decorator
def example():
"""Docstring"""
print('Called example function')
这些功能使得 functools 成为编写更加灵活和可维护的代码的有用工具。在函数式编程、装饰器和缓存等方面,functools 提供了一些实用的功能。
4、函数式编程的实际应用
4.1、函数式编程风格的代码
函数式编程风格的代码通常更简洁、清晰,具有更好的可读性。
代码语言:javascript复制# 命令式编程
result = []
for number in numbers:
result.append(number * 2)
# 函数式编程
result = map(lambda x: x * 2, numbers)
4.2、不可变性和线程安全性
函数式编程中的不可变性使得代码更容易在多线程环境中工作,因为不需要担心共享状态的问题。
代码语言:javascript复制# 命令式编程
shared_state = 0
def increment_state():
global shared_state
shared_state = 1
# 函数式编程
immutable_state = 0
def increment_state(state):
return state 1
5、小结
本文Python中函数式编程的基础概念、语法和实际应用。通过深入理解和应用函数式编程,你可以写出更具表达力、模块化和可维护性的代码。在实际项目中,根据需要灵活选择编程范式,将函数式编程的思想融入到你的Python代码中,发挥Python强大而灵活的特性。