python3学习笔记(9)_closure

 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())

 

posted @ 2017-07-13 17:20  wjwdive  阅读(379)  评论(0编辑  收藏  举报