1 #python 学习笔记 2017/07/13
2 # !/usr/bin/env python3
3 # -*- conding:utf-8 -*-
4
5 #从高阶函数的定义,我们可以知道,把函数作为参数的函数,把函数作为返回值的参数都是高阶函数
6
7 #可变参数求和
8 def calc_sum(*args):
9 sum = 0
10 for n in args:
11 sum = sum + n
12 return sum
13
14 #这个函数会立即返回求和结果,如果不要立即返回求和结果,而是需要在计算,可以返回求和函数,当再次调用时才给出计算结果
15 def lazy_sum(*args):
16 def sum():
17 tem = 0
18 for n in args:
19 tem = tem + n
20 return tem
21 return sum
22
23 #我们调用 lazy_sum()时,返回的并不是求和结果,而是求和函数
24 f = lazy_sum(1,2,3)
25 print("f addr:",f)
26
27 print("f() :",f())
28
29 #这种函数lazy_sum内部再定义一个函数,且内函数sum可以引用外部函数lazy_sum的参数和局部变量,当lazy_sum返回sum时,相关的参数和变量都保存在返回的函数中,这就是闭包。
30 #在Javascript中我也写过几篇理解闭包的学习笔记,Javascript的旧标准是没有作用域概念的,作用域用函数来区分,闭包就可以保持作用域
31 #延长变量的 存活周期,缺点也就随之而来,就是用的多的话内存会保留大量的变量,占用内存空间
32
33
34 #我们需要注意的是,每次调用 lazy_sum()时,都会返回一个全新的函数,即使传入的参数完全相同
35 # 如
36 f1 = lazy_sum(1,2)
37 f2 = lazy_sum(1,2)
38 print(f1 == f2)#False
39
40 #闭包里返回的函数没有立即执行,而是直接调用函数f()时,才执行
41 def count():
42 fs = []
43 for i in range(1, 4):
44 def f():
45 return i*i
46 fs.append(f)
47 return fs
48 f1,f2,f3 = count()
49 print("f1():", f1(), "f2() :", f2(), "f3():", f3())#9 , 9, 9
50 #发现 调用 f1(), f2() ,f3() 的结果全部是9, 这是因为返回的函数引用了变量i, 但它并非立刻执行。
51 #等到3个函数都返回时,他们所引用的变量 i 已经变成了3.最终结果是9
52
53 #所以返回闭包时,返回函数不要引用任何循环变量,或者后续会发生变换的变量
54
55 #如果一定要引用循环变量,可以再创建一个函数,用该函数的 参数绑定循环变量当前的值,无论该循环变量后续如何更改,已经绑定到函数参数的值不会变
56 def countNew():
57 def f(j):
58 def g():
59 return j*j
60 return g
61 fs = []
62 for i in range(1, 4):
63 fs.append(f(i))#f(i)被立即执行,因此,i的当前值被传入f()
64 return fs
65
66 f1, f2, f3 = countNew()
67 print(f1(), f2(), f3())