• 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())
  • 相关阅读:
    Golang之字符串格式化
    BZOJ 4513: [Sdoi2016]储能表 [数位DP !]
    BZOJ 3329: Xorequ [数位DP 矩阵乘法]
    BZOJ 1833: [ZJOI2010]count 数字计数 [数位DP]
    HDU2089 不要62 BZOJ1026: [SCOI2009]windy数 [数位DP]
    未完
    [Miller-Rabin & Pollard-rho]【学习笔记】
    BZOJ 3551: [ONTAK2010]Peaks加强版 [Kruskal重构树 dfs序 主席树]
    BZOJ 3123: [Sdoi2013]森林 [主席树启发式合并]
    BZOJ 3545: [ONTAK2010]Peaks [Splay启发式合并]
  • 原文地址:https://www.cnblogs.com/wjw-blog/p/7161599.html
Copyright © 2020-2023  润新知