主要是跟着廖老师的学习教材学习
函数式编程FP
函数是把一段能够实现某种特定功能的代码取了一个名字,在这之后调用这个名字即可实现功能的一种封装。把一个复杂任务通过层层函数调用简单化。
函数式编程——Functional Programming能够调用多个函数来实现功能。
函数式编程是一种抽象程度很高的编程范式,纯粹的函数式编程语言编写的函数没有变量,而python允许使用变量,因此python不是纯函数式编程语言。
高阶函数
函数名也是变量,若把一些内置函数定义成数值,那么就会丧失原有功能。
>>> str = 10
>>> str(-10)
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
TypeError: 'int' object is not callable
可以使用del str来恢复pow函数。当然最好还是重启python交互环境。
函数式编程的参数可以为函数`
def add(x, y, f):
return f(x) + f(y)
print(add(-5, 6, str))
-56
map/reduce
map
map() 函数语法:
map(function, iterable)
map作用在于把iterable内的每个元素都调用一次function,返回每次function返回值组成的iterator.
map接收两个参数,一个是函数,另一个是可迭代对象(比如list,tuple),map结果返回一个iterator(惰性序列),因此我们可以通过list()函数来把整个序列计算出来返回一个list,也可以巧妙运用list的append方法生成一个新的list.
reduce
reduce() 函数语法:
reduce(function, iterable[, initializer])
initializer为可选参数。
有别于map的是
1.reduce中的function必须接受两个参数,把结果当做新参数再与序列下一个元素放入function进行计算。最简单的例子就是sum除了内建函数还可以这样实现
from functools import reduce
def add(x,y):
return x+y
print(reduce(add,range(0,10)))
通过匿名函数lambda,代码还能进一步简化
from functools import reduce
print(reduce(lambda x,y:x+y,range(0,10)))
filter
filter()也就是过滤函数,与map一样也接受一个函数和序列,也返回一个Iterator
案例:筛选1到200的回数(从左往右和从右往左都是一样的数)
def huishu(n))
return n==int(str(n)[::-1])
sorted
sorted()是排序函数,可以接受一个key函数来自定义排序,key作用于每一个字符。
案例:对一组表示学生名字和成绩的tuple按照名字排序以及按成绩排序。
L = [('Bob', 75), ('Adam', 92), ('Bart', 66)]
def by_name(t):
return t[0]
def by_grade(h):
return h[-1]
L2 = sorted(L, key=by_name)
L3 = sorted(L, key=by_grade)
返回函数
高阶函数可以把函数作为返回值。
比如定义一个求和函数:
>>> def calc_sum(num_list):
s = 0
for i in num_list:
s += i
return s
倘若我们不需要立即求和,而是放在后面代码中根据需要再计算(ps:在我看来相对于一个小括号包裹了该函数),我们可以返回求和的函数而非计算结果。
>>> def lazy_calc_sum(num_list):
def calc_sum():
s = 0
for i in num_list:
s += i
return s
return calc_sum
>>> f_lazy = lazy_calc_sum([1,2,3,4])
>>> f_lazy
<function lazy_calc_sum.<locals>.calc_sum at 0x0000003B8D25A2D8>
>>> f_lazy()
10
在这之中调用lazy_calc_sum()返回的是calc_sum(),唯有调动f_lazy时才是真正计算结果。
注:哪怕实例调用同一个函数,且参数一样,它的返回函数都是一个新的函数。也就是
>>> f1 = lazy_calc_sum([1, 3, 5, 7, 9])
>>> f2 = lazy_calc_sum([1, 3, 5, 7, 9])
>>> f1==f2
False
>>>print(id(f1))
140496661629400
>>>print(id(f2))
140496433033288
闭包
关于闭包的知识参考了一位博主的文章
链接:原文
如果一个函数定义中引用了函数外定义的变量,并且该函数可以在其定义环境外被执行。这样的一个函数我们称之为闭包。
示例:
def outer():
a = 6
def inner():
b = 8
print(a)
print(b)
return inner
if __name__ == '__main__':
res = outer()
res()
执行结果:
>>>6
>>>8
自由变量a作为outer()的局部变量被内部函数引用。而当a在内部函数以局部变量定义时,就不存在闭包了
def outer():
a = 6
def inner():
b = 8
a += 1
print(a)
print(b)
return inner
if __name__ == '__main__':
res = outer()
res()
执行函数会报错:UnboundLocalError: local variable 'a' referenced before assignment
局部变量在某种程度上优先级是大于全局变量的。
匿名函数
- Python用lambda语法定义匿名函数
- lambda只需用表达式而无需申明
- lambda没有函数名
- 在定义时直接被调用,冒号后面的表达式有且只能有一个
- 自带return
装饰器
本质上,decorator是一个返回函数的高阶函数。在代码运行期间增加其功能,不会修改该函数定义。
定义装饰器的时候用def,调用的时候须用@.python内置的functools.wraps装饰器可以把原始函数的__name__等属性复制到wrapper()函数中。
案例:设计dectorator,打印该函数执行时间。
import time
import functools
def metric(func):
@functools.wraps(func)
def wrapper(*args, **kw):
start = time.time()
func(*args, **kw) # 需要先运行一下函数,记录开始时间和结束时间。
end = time.time()
print('%s executed in %s ms' % (func.__name__, (end - start)))
return func(*args, **kw)
return wrapper
@metric
def fast(x, y):
time.sleep(0.0012)
return x + y;
@metric
def slow(x, y, z):
time.sleep(0.1234)
return x * y * z;
f = fast(11, 22)
s = slow(11, 22, 33)
if f != 33:
print('测试失败!')
elif s != 7986:
print('测试失败!')
偏函数
此偏函数不等同于数学上的偏函数,用于把一个函数的某些参数设定默认值,返回一个新的函数。
案例:
import functools
sorted2 = functools.partial(sorted,key=abs)
print(sorted2([1,-3,-4]))