参考:https://www.cnblogs.com/mzc1997/p/7641995.html

  Python中函数不仅可以作为参数还可以作为结果返回

>>> def pro1(c,f):
... def pro2():
... return f(c)
... return pro2
#调用函数pro1函数时,返回的是pro2函数对象
>>> a=pro1(-3,abs)
>>> a
<function pro1.<locals>.pro2 at 0x7fe07ed4eea0>
#需要对a调用才能得到结果
>>> a()
3 #执行了a()相当于f(c)执行了 abs(-3)返回为3

  如果在一个内部函数里对外部作用域(但不是全局作用域)的变量进行引用,内部函数称为闭包(closure)

  

>>> def fun1(n):
... def fun2(x):
... return pow(x,n)
... return fun2
...
>>> pow2=fun1(2)
>>> pow2
<function fun1.<locals>.fun2 at 0x7f8bdfe2e378>
>>> pow2(9)
81

  函数fun2对fun1的参数n进行了引用,将带参数的fun1给一个新的函数pow2,新的函数再加一个参数x运行,则相当于pow(9,2) 9的二次方

  当fun1的生命周期结束时,已经引用的变量n存在fun2中,依然可以调用

  在看一个例子count.py

def count():
fs=[]
for i in range(1,4):
def f():
return i*i
#把函数f地址追加至列表fs
fs.append(f)
#返回为一个包含3个函数f地址的列表
return fs print('count函数返回一个列表,列表分别为函数f地址',count())
f1,f2,f3 = count()
print(f1,'f1执行结果是',f1())
print(f2,'f2执行结果是',f2())
print(f3,'f3执行结果是',f3())

  运行结果

count函数返回一个列表,列表分别为函数f地址 [<function count.<locals>.f at 0x7fc5d30cac80>, <function count.<locals>.f at 0x7fc5d30cad08>, <function count.<locals>.f at 0x7fc5d30cad90>]
<function count.<locals>.f at 0x7fc5d30cac80> f1执行结果是 9
<function count.<locals>.f at 0x7fc5d30cad08> f2执行结果是 9
<function count.<locals>.f at 0x7fc5d30cad90> f3执行结果是 9

  结果全是9不是预期的1,4,9

  结果全部都是9. 不是预期的1,4,9。
  原因是返回函数引用了变量i,下面来解析一下f1,f2,f3=count()这句的执行过程:
   当i=1, 执行for循环, 结果返回函数f的函数地址,存在列表fs中的第一个位置上。
   当i=2, 由于fs列表中第一个元素所指的函数中的i是count函数的局部变量,i也指向了2;然后执行for循环, 结果返回函数f的函数地址,存在列表fs中的第二个位置上。
   当i=3, 同理,在fs列表第一个和第二个元素所指的函数中的i变量指向了3; 然后执行for循环, 结果返回函数f的函数地址,存在列表fs中的第三个位置上。
  所以在调用f1()的时候,函数中的i是指向3的:
    f1():
       return 3*3
  同理f2(), f3()结果都为9

  

  闭包时牢记的一点就是:返回函数不要引用任何循环变量,或者后续会发生变化的变量。即包在里面的函数(本例为f()),不要引用外部函数(本例为count())的任何循环变量

  如果一定要引用循环变量怎么办?方法是再创建一个函数,用该函数的参数绑定循环变量当前的值,无论该循环变量后续如何更改,已绑定到函数参数的值不变:

  count_2.py

def count():
fs=[]
for i in range(1,4):
def f(j):
def g():
return j*j
return g
fs.append(f(i))
#返回为一个包含3个函数f地址的列表
return fs print('count函数返回一个列表,列表分别为函数f地址',count())
f1,f2,f3 = count()
print(f1,'f1执行结果是',f1())
print(f2,'f2执行结果是',f2())
print(f3,'f3执行结果是',f3())

  

count函数返回一个列表,列表分别为函数f地址 [<function count.<locals>.f.<locals>.g at 0x7ff488311d08>, <function count.<locals>.f.<locals>.g at 0x7ff488311c80>, <function count.<locals>.f.<locals>.g at 0x7ff488311d90>]
<function count.<locals>.f.<locals>.g at 0x7ff488311c80> f1执行结果是 1
<function count.<locals>.f.<locals>.g at 0x7ff488311d08> f2执行结果是 4
<function count.<locals>.f.<locals>.g at 0x7ff488311d90> f3执行结果是 9

  结果就是预期的1,4,9.
  当i=1时,f(1)即让j指向1,
  当i=2时,f(2)即让j指向2,此时j不是count的局部变量,不会影响到i=1是f(1)中j的指向。即函数f的参数绑定循环变量当前的值, 而不是循环变量本身。

Python3之返回函数的更多相关文章

  1. Python3返回函数

    函数作为返回值 高阶函数除了可以接受函数作为参数外,还可以把函数作为结果值返回. 我们来实现一个可变参数的求和.通常情况下,求和的函数是这样定义的: def calc_sum(*args): ax = ...

  2. [Python3] 036 函数式编程 返回函数

    目录 函数式编程 之 返回函数 1. 引子 2. 闭包 closure 函数式编程 之 返回函数 函数可以返回具体的值 也可以返回一个函数作为结果 1. 引子 1.1 定义一个普通函数 >> ...

  3. python3之内置函数

    1.abs() 取数字的绝对值 >>> print(abs(-28)) 28 >>> print(abs(-2.34)) 2.34 >>> pri ...

  4. Python3基础-特别函数(map filter partial reduces sorted)实例学习

    1. 装饰器 关于Python装饰器的讲解,网上一搜有很多资料,有些资料讲的很详细.因此,我不再详述,我会给出一些连接,帮助理解. 探究functools模块wraps装饰器的用途 案例1 impor ...

  5. python3内置函数大全(顺序排列)

    python3内置函数大全 内置函数 (1)abs(),   绝对值或复数的模 1 print(abs(-6))#>>>>6 (2)all() 接受一个迭代器,如果迭代器的所有 ...

  6. Python3中map函数的问题

    在Python2中map函数会返回一个list列表,如代码: >>> def f(x, y): return (x, y) >>> l1 = [ 0, 1, 2, ...

  7. Python3.x:函数定义

    Python3.x:函数定义 1,函数定义: def 函数名称([参数1,参数2,参数3......]): 执行语句 2,实例一(不带参数和没返回值): def helloWorld(): print ...

  8. Python学习札记(二十四) 函数式编程5 返回函数

    参考:返回函数 NOTE 1.高阶函数除了可以接受函数作为参数外,还可以把函数作为结果值返回. eg.求和函数 #!/usr/bin/env python3 def calsums(*args): a ...

  9. python3内置函数大全

    由于面试的时候有时候会问到python的几个基本内置函数,由于记不太清,就比较难受,于是呕心沥血总结了一下python3的基本内置函数 Github源码:        https://github. ...

随机推荐

  1. Linux centOS 6 和 centOS 7一些简单的区别

    最明显的是,获取网络端口的命令和IP信息的命令不同了 所以刚安装的centos 7 使用ifconfig查看ip时有可能出现-bash: ifconfig: 未找到命令

  2. CF843D Dynamic Shortest Path spfa+剪枝

    考试的T3,拿暴力+剪枝卡过去了. 没想到 CF 上也能过 ~ code: #include <bits/stdc++.h> #define N 100004 #define LL lon ...

  3. QLocalSocket

    QIODevice做为QLocalSocket的父类 在Qt中,提供了多种IPC方法.看起来好像和Socket搭上点边,实则底层是windows的name pipe.这应该是支持双工通信的 QLoca ...

  4. luogu 2152

    SuperGcd 二进制算法 1. A = B, Gcd(A, B) = A; 2. A,B为偶数,  Gcd(A, B) = 2 * Gcd(A / 2, B / 2); 3. A 为偶数, B 为 ...

  5. 02_Kibana的安装

    Kibana部署 1. 下载Kibana $ wget https://artifacts.elastic.co/downloads/kibana/kibana-6.6.0-linux-x86_64. ...

  6. ubuntu ukylin wineqq 登录时提示:您的号码暂时不能使用低版本的qq

    ubuntu ukylin wineqq 登录时提示:您的号码暂时不能使用低版本的qq,而有的qq号登录没有问题. 优麒麟官网上下载的qqwine安装包,解压后安装三个deb包. 郁闷了一下午,都想装 ...

  7. 配置Notepad++

    Notepad++配置 1.自动换行 视图 - 自动换行 2.隐藏工具栏 设置 - 首选项... > 常用 > 工具栏 - 隐藏 3.隐藏菜单栏 设置 - 首选项... > 常用 & ...

  8. python常用函数2

    2.reduce()函数 reduce() 函数也是python内置的一个高阶函数.reduce()函数接收的参数和 map()相似,一个函数   f ,一个list,但行为和  map()不同,re ...

  9. C++标准库分析总结(三)——<迭代器设计原则>

    本节主要总结迭代器的设计原则,以及iterstor traits的设计作用 1.迭代器遵循的原则 迭代器是算法和容器的桥梁,它是类模板的设计,迭代器必须有能力回答算法提出的问题才能去搭配该算法的使用 ...

  10. 手写一个MVVM

    最近看了珠峰的架构课——实现一个MVVM. 首先,我们来了解一下什么是MVVM. MVVM是Model-View-ViewModel的简写.它本质上就是MVC 的改进版.MVVM 就是将其中的View ...