详解高阶函数和闭包 | 手把手教你入门Python之四十五

简介: 本节重点介绍高阶函数 ,闭包

上一篇:递归函数和匿名函数的使用介绍 | 手把手教你入门Python之四十四
下一篇:5个案例详解装饰器 | 手把手教你入门Python之四十六

本文来自于千锋教育在阿里云开发者社区学习中心上线课程《Python入门2020最新大课》,主讲人姜伟。

高阶函数

在Python中,函数其实也是⼀种数据类型。

def test():
 return 'hello world'
print(type(test)) # <class 'function'>

函数对应的数据类型是 function ,可以把它当做是⼀种复杂的数据类型。
既然同样都是⼀种数据类型,我们就可以把它当做数字或者字符串来处理。

定义⼀个变量指向函数

在Python中,我们还可以定义⼀个变量,让它来指向⼀个函数,相当于给函数起了⼀个别名。

def test():
 return 'hello wrold'
fun = test # 定义了⼀个变量fun,让它指向了 test 这个函数
print(fun()) # 使⽤fun()可以直接调⽤test这个函数
print(id(fun)) # 1819677672040
print(id(test)) # 1819677672040

注意:在定义⼀个变量表示⼀个函数时,函数后⾯不能加括号!加括号表示的是调⽤这个函数。

def test():
 return 'hello world'
result = test() # 这种写法是调⽤test函数,并把函数的返回值赋值给result变量
print(result()) # 这⾥会报错 TypeError: 'str' object is not callable
fun = test # 这种写法是给test函数起了⼀个别名,注意,这⾥的test后⾯不能加()
fun() # 可以使⽤别名调⽤这个函数

⾼阶函数

既然变量可以指向函数,函数的参数能接收变量,那么⼀个函数就可以接收另⼀个函数作为参数,同样,我们还可以把⼀个函数当做另⼀个函数的返回值。这种函数的使⽤⽅式我们称之为⾼阶函数。

函数做为另⼀个函数的参数

def test(age,action):
 if age < 18:
 print('您还没满⼗⼋岁,请退出')
 action() # 把参数action直接当做⼀个函数来调⽤
def smoke():
 print('我已经年满⼗⼋岁了,我想抽烟')

my_action = smoke # 定义⼀个变量my_action,让它指向smoke函数
test(21, my_action) # 将my_action传给 test 函数作为它的参数
test(21,smoke) # 还可以不再定义⼀个新的变量,直接传⼊函数名

函数作为另⼀个函数的返回值

def test():
 print('我是test函数⾥输⼊的内容')
def demo():
 print('我是demo⾥输⼊的内容')
 return test # test 函数作为demo函数的返回值
result = demo() # 我是demo⾥输⼊的内容 调⽤ demo 函数,把demo函数的返回值赋值给 result
print(type(result)) # <class 'function'> result 的类型是⼀个函数
result() # 我是demo⾥输⼊的内容 我是test函数⾥输⼊的内容 既然result是⼀个函数,那么就可
以直接使⽤() 调⽤这个函数
demo()() # 我是demo⾥输⼊的内容 我是test函数⾥输⼊的内容

image.png
image.png
image.png

image.png

闭包

函数只是⼀段可执⾏代码,编译后就“固化”了,每个函数在内存中只有⼀份实例,得到函数的⼊⼝点便可以执⾏函数了。函数还可以嵌套定义,即在⼀个函数内部可以定义另⼀个函数,有了嵌套函数这种结构,便会产⽣闭包问题。

函数嵌套

在函数⾥⾯还可以定义函数,可以嵌套多层,执⾏需要被调⽤。

def outer():
 print('outer----hello')
 def inner(): # inner这个函数是在outer函数内部定义的
 print('inner----hello')
 inner() # inner函数只在outer函数内部可⻅
outer()
# inner() 这⾥会报错,在outer函数外部⽆法访问到inner函数

什么是闭包

闭包是由函数及其相关的引⽤环境组合⽽成的实体(即:闭包=函数块+引⽤环境)。

def outer(n):
 num = n
 def inner():
 return num+1
 return inner
print(outer(3)()) # 4
print(outer(5)()) # 5

在这段程序中,函数 inner 是函数 outer 的内嵌函数,并且 inner 函数是outer函数的返回值。我们注意到⼀个问题:内嵌函数 inner 中引⽤到外层函数中的局部变量num,Python解释器会这么处理这个问题呢? 先让我们来看看这段代码的运⾏结果,当我们调⽤分别由不同的参数调⽤ outer 函数得到的函数时,得到的结果是隔离的(相互不影响),也就是说每次调⽤outer函数后都将⽣成并保存⼀个新的局部变量num,这⾥outer函数返回的就是闭包。 如果在⼀个内部函数⾥,对在外部作⽤域(但不是在全局作⽤域)的变量进⾏引⽤,那么内部函数就被认为是闭包(closure)。

修改外部变量的值

闭包⾥默认不能修改外部变量。

def outer(n):
 num = n
 def inner():
 num = num + 1
 return num
 return inner
print(outer(1)())

上述代码运⾏时会报错!

UnboundLocalError: local variable 'num' referenced before assignment

原因分析

在python⾥,只要看到了赋值语句,就会认为赋值语句的左边是⼀个局部变量。 num = num + 1 这段代码⾥, num 在 = 的左边,python解析器会认为我们要修改 inner 函数⾥ num 这个局部变量,⽽这个变量使⽤之前是未声明的,所以会报错。

解决方案

我们分析过,报错的原因在于当我们在闭包内修改外部变量时,会被python解析器误会为内部函数的局部变量。所以,解决⽅案就在于,我们需要想办法,让解析器知道我们不是要修改局部变量,⽽是要修改外部变量。

  • 解决⽅法:使⽤ nonlocal 关键字
def outer(n):
 num = n
 def inner():
 nonlocal num # 修改前使⽤nonlocal关键字对 num 变量进⾏说明
 num = num + 1
 return num
 return inner
print(outer(2)())

image.png

计算代码的执行时长

image.png
image.png

相关文章
|
6月前
|
SQL 关系型数据库 数据库
Python SQLAlchemy模块:从入门到实战的数据库操作指南
免费提供Python+PyCharm编程环境,结合SQLAlchemy ORM框架详解数据库开发。涵盖连接配置、模型定义、CRUD操作、事务控制及Alembic迁移工具,以电商订单系统为例,深入讲解高并发场景下的性能优化与最佳实践,助你高效构建数据驱动应用。
732 7
|
7月前
|
测试技术 开发者 Python
Python单元测试入门:3个核心断言方法,帮你快速定位代码bug
本文介绍Python单元测试基础,详解`unittest`框架中的三大核心断言方法:`assertEqual`验证值相等,`assertTrue`和`assertFalse`判断条件真假。通过实例演示其用法,帮助开发者自动化检测代码逻辑,提升测试效率与可靠性。
513 1
|
6月前
|
Cloud Native 算法 API
Python API接口实战指南:从入门到精通
🌟蒋星熠Jaxonic,技术宇宙的星际旅人。深耕API开发,以Python为舟,探索RESTful、GraphQL等接口奥秘。擅长requests、aiohttp实战,专注性能优化与架构设计,用代码连接万物,谱写极客诗篇。
1217 1
Python API接口实战指南:从入门到精通
|
6月前
|
存储 Java 调度
Python定时任务实战:APScheduler从入门到精通
APScheduler是Python强大的定时任务框架,通过触发器、执行器、任务存储和调度器四大组件,灵活实现各类周期性任务。支持内存、数据库、Redis等持久化存储,适用于Web集成、数据抓取、邮件发送等场景,解决传统sleep循环的诸多缺陷,助力构建稳定可靠的自动化系统。(238字)
1091 1
|
7月前
|
调度 数据库 Python
Python异步编程入门:asyncio让并发变得更简单
Python异步编程入门:asyncio让并发变得更简单
394 5
|
7月前
|
数据采集 存储 XML
Python爬虫入门(1)
在互联网时代,数据成为宝贵资源,Python凭借简洁语法和丰富库支持,成为编写网络爬虫的首选。本文介绍Python爬虫基础,涵盖请求发送、内容解析、数据存储等核心环节,并提供环境配置及实战示例,助你快速入门并掌握数据抓取技巧。
|
7月前
|
大数据 数据处理 数据安全/隐私保护
Python3 迭代器与生成器详解:从入门到实践
简介:本文深入解析Python中处理数据序列的利器——迭代器与生成器。通过通俗语言与实战案例,讲解其核心原理、自定义实现及大数据处理中的高效应用。
343 0
|
7月前
|
存储 缓存 安全
Python字典:从入门到精通的实用指南
Python字典如瑞士军刀般强大,以键值对实现高效数据存储与查找,广泛应用于配置管理、缓存、统计等场景。本文详解字典基础、进阶技巧、实战应用与常见陷阱,助你掌握这一核心数据结构,写出更高效、优雅的Python代码。
183 0
|
8月前
|
数据挖掘 数据处理 C++
Python Lambda:从入门到实战的轻量级函数指南
本文通过10个典型场景,详解Python中Lambda匿名函数的用法。Lambda适用于数据处理、排序、条件筛选、事件绑定等简洁逻辑,能提升代码简洁性和开发效率。同时提醒避免在复杂逻辑中过度使用。掌握Lambda,助你写出更高效的Python代码。
506 0
|
8月前
|
数据采集 Web App开发 JSON
Python爬虫基本原理与HTTP协议详解:从入门到实践
本文介绍了Python爬虫的核心知识,涵盖HTTP协议基础、请求与响应流程、常用库(如requests、BeautifulSoup)、反爬应对策略及实战案例(如爬取豆瓣电影Top250),帮助读者系统掌握数据采集技能。
673 0

推荐镜像

更多