Python基础之函数:2、globlal与nonlocal和闭包函数、装饰器、语法糖
一、global与nonlocal
1、global
- 在py文件中,一般无法调用函数体内变量名,而global可使函数体代码内的变量名直接在函数体外部调用,条件是在需要调用的代码体中使用global 调用需要的变量名
未使用global情况:
# 在外部绑定一个变量名
name = 'kangkng'
# 定义一个函数体代码
def func():
# 函数体内部重新绑定一个变量名
name = 'zhangzhang'
# 调用函数func
func()
# 这时无法打印函数体内部的变量名
print(name)
------------------------------------------------------------------------ kangkang
使用global情况:
# 在函数体内部使用global时,可在外部直接调用函数内部变量名
name = 'kangkng'
# 定义一个函数体代码
def func():
# 使用global调用变量名
global name
# 函数体内部重新绑定一个变量名
name = 'zhangzhang'
# 调用函数func
func()
# 外py中打印name
print(name)
------------------------------------------------------------------------
zhangzhang
2、nonlocal
- 在函数嵌套使用时,通常在父代码体中无法调用子代码中变量名,
而nonlocal的作用是,可以在父代码中直接调用子代码中的变量名,条件是需要在子代码中使用nonlocal 调用需要的变量名
未使用nonlocal情况:
# 定义一个函数体代码
def outer():
# 绑定一个变量名
name = 'kangkang'
# 代码体内部再次定义一段函数体
def subcoat():
# 内层中绑定变量名
name = 'zhangzhang'
# 在函数外层打印变量名
print(name)
# 调用外层函数体代码
outer()
-----------------------------------------------------------------------
kangkang
使用nonlocal情况:
# 在函数体内部使用global时,可在外部直接调用函数内部变量名
def outer():
# 函数外层绑定一个变量名
name = 'kangkang'
# 代码体内部再次定义一段函数体
def subcoat():
# 在函数体内层使用nonlocal,调用变量名
nonlocal name
# 内层中绑定变量名
name = 'zhangzhang'
# 调用内层函数
subcoat()
# 在函数外层打印变量名
print(name)
# 调用外层函数体代码
outer()
----------------------------------------------------------------------
zhangzhang
二、函数名的多种用法
引言:
函数名就相当于变量名,只不过函数名绑定的是一段函数体代码,在我们使用这个函数名加括号时就可以调用这段代码体,具体由以下几种用法:
1、当做变量名赋值
def func():
print('我是func函数体代码')
res = func
print(res())
------------------------------------------------------------------------
我是func函数体代码
None
2、当作函数的参数
def func():
print('我是func函数体代码')
def func1(a):
print('我是func1函数体代码', a)
a()
func1(func)
------------------------------------------------------------------------
我是func1函数体代码 <function func at 0x000001D0C14D6310>
我是func函数体代码
3、当作函数的返回值
def func():
print('我是func函数体代码')
def func1():
print('我是func1函数体代码')
return func
res = func1()
print(res)
res()
------------------------------------------------------------------------
我是func1函数体代码
<function func at 0x00000218F95B6310>
我是func函数体代码
4、当作容器类型的数据
def spring():
print('我是春季,生机盎然')
def summer():
print('我是夏季,活力四射')
def autumn():
print('我是秋季,翩翩起舞')
def winter():
print('我是冬季,大雪纷飞')
while True:
season_dict = { '1': spring,
'2': summer,
'3': autumn,
'4': winter
}
season_select = input('根据编号,选择您喜欢的季节>>>:').strip()
if season_select in season_dict:
season_dict.get(season_select)()
else:
print('你选择的编号不存在')
------------------------------------------------------------------------
三、闭包函数
1、什么是闭包函数
一个函数的返回值是另外一个函数,返回的函数调用父函数内部的变量,如果返回的函数在外部被执行,就产生了闭包
2、闭包函数需满足的条件
满足以下两个条件的就是闭包函数:
条件一:
定义在函数内部的函数
条件二:
用到了外部函数空间名称中的名子
3、闭包函数的作用
作用:
使函数外部能够调用函数内部放入属性和方法
缺点:
闭包操作导致整个函数的内部环境被长久保存,占用大量内存
4、闭包函数的实际应用
1.函数内部变量名在外部被访问
def fun1():
name = 'python'
def inner():
print(name)
return inner
result = fun1()
result()
------------------------------------------------------------------------
python
2.函数体内部函数体代码可以通过外部访问
def fun2():
def inner():
print("执行了内部函数inner")
def all():
return inner
return all
result = fun2()
result()()
------------------------------------------------------------------------
执行了内部函数inner
四、装饰器
当我们需要将一段函数体代码在不改变调用方式和源代码的情况下,需要给这个段代码添加新的功能时,这时候我们就需要给这段代码安装一个装饰器,装饰器是指将这段代码封装在闭包函数内,来达到既能满足上述条件,又能增加新的功能的条件
概念
- 在不修改被装饰对象源代码和调用方式的情况下给被装饰的对象添加新的功能
本质
- 并不是一门新的技术,而是由函数参数、名称空间、函数名多种用法、闭包函数组合到一起的效果
口诀
- 对修改封闭,对扩展开放
1、装饰器推导流程
1、首先定义一段函数体代码,当我们给这段函数传入指定的参数时,他就会暂停一秒,然后运行,使它在运行结束后,能够统计它的运行时间
import time
def index(a, b):
time.sleep(1)
print(index,a, b)
2、通常,我们只需要在这段代码运行前打印一个时间戳,运行后再次打印一个时间戳,在这段代码运行结束后通过前后时间的插值就能统计出这段代码的运行时间,但这种办法使用起来比较麻烦且只能使用一次
方法一:
import time
def index(a, b):
start = time.time()
time.sleep(1)
print(index, a, b)
end = time.time()
print(end - start)
index(1,2)
方式二:
import time
def index(a, b):
time.sleep(1)
print(index, a, b)
start = time.time()
index(1,2)
end = time.time()
print(end - start)
3、通过上述方法的方式二,我们可以得出将函数名包裹在统计时间功能代码内,这样在调用时相对便捷,进一步思考,若将这段代码使用函数封包,那样在调用时就可以更为便捷,在以后统计该代码时,只需要调用封包这段代码的函数名就可以直接统计这段代码的运行时间
import time
def index(a, b):
time.sleep(1)
print(index, a, b)
def time_():
start = time.time()
index()
end = time.time()
print(end - start)
time_()
------------------------------------------------------------------------
Traceback (most recent call last):
File "D:/pytcharm项目文件路径/38/11.py", line 297, in <module>
time_()
File "D:/pytcharm项目文件路径/38/11.py", line 293, in time_
index()
TypeError: index() missing 2 required positional arguments: 'a' and 'b'
4、虽然这种方式可以行得通,但只能针对没有参数的函数体代码,若这段代码需要传参者无法运行,并直接报错。再次进一步思考,只需要将封包的这段函数设置为有参函数就可解决这个问题
import time
def index(a, b):
time.sleep(1)
print(index, a, b)
def core(a,b):
start = time.time()
index(a, b)
end = time.time()
print(end - start)
core(1, 2)
------------------------------------------------------------------------
<function index at 0x000001F4A0026310> 1 2
1.0047826766967773
5、由上推导可看出,虽然此功能可以更为便捷的统计代码执行时间,但若是源代码的参数需要修改则封包它的参数也需要修改,这时我们可联想到将参数修改为可变长参数,就不会出现这个问题
import time
def index(a, b):
time.sleep(1)
print(index, a, b)
def core(*args,**kwargs):
start = time.time()
index(*args, **kwargs)
end = time.time()
print(end - start)
core(1,2)
------------------------------------------------------------------------
<function index at 0x000002ECDD4E6310> 1 2
1.004744529724121
6、这样无论源代码参数如何修改,我们都可以进行传参,虽然这个问题解决了,但考虑使用的广泛性,若有其他函数体也需要用到这个功能时,还需要重新修改封包内代码,这时,我们可以使用闭包的方式来满足这个条件
import time
def index(a, b):
time.sleep(1)
print(index, a, b)
def func(x, y, z):
time.sleep(2)
print(func, x, y, z)
def outer(index):
def core(*args, **kwargs):
start = time.time()
index(*args, **kwargs)
end = time.time()
print(end - start)
return core
res = outer(func)
res(1, 2, 3)
------------------------------------------------------------------------
<function func at 0x0000018C23686670> 1 2 3
2.00856614112854
7、通过将源代码函数名放至闭包函数参数内,就可以达到可以调动任何函数体代码都可以执行此功能的方法,但并未满足闭包函数的条件,源代码的调用方式改变了,这时我们可以通过将原函数体代码赋值的方式来达到调用方式和源代码都未改变的情况下来增加此功能
import time
def index(a, b):
time.sleep(1)
print(index, a, b)
def func(x, y, z):
time.sleep(2)
print(func, x, y, z)
def outer(index):
def core(*args, **kwargs):
start = time.time()
index(*args, **kwargs)
end = time.time()
print(end - start)
return core
index = outer(index)
index(1,2)
func = outer(func)
func(1, 2, 3)
------------------------------------------------------------------------
<function outer.<locals>.core at 0x0000026C17F58280> 1 2
1.004807710647583
<function outer.<locals>.core at 0x0000026C17F58940> 1 2 3
2.0077626705169678
8、虽然上述推导过程都已满足装饰器条件,但是考虑到源代码有返回值的情况,我们没有并没有获取,这时在进一步推导,可在装饰器函数内部调用源代码函数名的位置设置一个变量名用于接收返回值,传给装饰器底层return用于接收即可解决这个问题
import time
def index(a, b):
time.sleep(1)
print(index, a, b)
return 'index'
def func(x, y, z):
time.sleep(2)
print(func, x, y, z)
return 'func'
def outer(index):
def core(*args, **kwargs):
start = time.time()
res = index(*args, **kwargs)
end = time.time()
print(end - start)
return res
return core
index = outer(index)
res = index(1,2)
print(res)
func = outer(func)
res = func(1, 2, 3)
print(res)
------------------------------------------------------------------------
<function outer.<locals>.core at 0x0000020C50A78280> 1 2
1.0050580501556396
index
<function outer.<locals>.core at 0x0000020C50A78940> 1 2 3
2.0094454288482666
func
2、装饰器语法糖
什么是装饰器语法糖
当我们使用装饰器调用被装饰的函数体代码时,总是需要在调用前通过赋值的方式来调用,这样的方式相对比较麻烦,这时我们就可以用到装饰器语法糖来节省时间和代码
语法糖的使用方法和条件
用法:
在源代码函数体上方使用@加装饰器函数名条件:
源代码需在装饰器下方
具体用法
import time
def outer(index):
def core(*args, **kwargs):
start = time.time()
res = index(*args, **kwargs)
end = time.time()
print(end - start)
return res
return core
@outer
def index(a, b):
time.sleep(1)
print(index, a, b)
return 'index'
index(1,2)
3、装饰器模板
def outer(func):
def inner(*args, **kwargs):
# 执行被装饰对象之前可以做的额外操作
res = func(*args, **kwargs)
# 执行被装饰对象之后可以做的额外操作
return res
return inner
Python基础之函数:2、globlal与nonlocal和闭包函数、装饰器、语法糖的更多相关文章
- Fluent_Python_Part3函数即对象,07-closure-decoration,闭包与装饰器
第7章 函数装饰器和闭包 装饰器用于在源码中"标记"函数,动态地增强函数的行为. 了解装饰器前提是理解闭包. 闭包除了在装饰器中有用以外,还是回调式编程和函数式编程风格的基础. 1 ...
- Python编程四大神兽:迭代器、生成器、闭包和装饰器
生成器 生成器是生成一个值的特殊函数,它具有这样一个特点:第一次执行该函数时,先从头按顺序执行,在碰到yield关键字时该函数会暂停执行该函数后续的代码,并且返回一个值:在下一次调用该函数执行时,程序 ...
- python装饰器 语法糖
简介: 装饰器(Decorators)是 Python 的一个重要部分.简单地说:他们是修改其他函数的功能的函数. 比如说我们写flask,路由就是用装饰器定义的.如果写权限控制,那么权限控制一般也是 ...
- python 装饰器(语法糖)
def login(func): def testlogin(): for i in range(3): _username="abc" ...
- Python记录9:函数4:名称空间作用域+闭包函数+装饰器
''' 一: 名称空间namespaces 名称空间就是存放名字与值绑定关系的内存空间 二: 名称空间分为三种 内置名称空间: 1. 特点: 存放是python解释器自 ...
- python_函数名的应用、闭包、装饰器
0.动态传参内容补充: 0.1 单纯运行如下函数不会报错. def func1(*args,**kwargs): pass func1() 0.2 *的魔性用法 * 在函数定义的时候,代表聚合. *在 ...
- python基础知识13---函数对象、函数嵌套、名称空间与作用域、装饰器
阅读目录 一 函数对象 二 函数嵌套 三 名称空间与作用域 四 闭包函数 五 装饰器 六 练习题 一 函数对象 1 函数是第一类对象,即函数可以当作数据传递 #1 可以被引用 #2 可以当作参数传递 ...
- 《python基础教程(第二版)》学习笔记 函数(第6章)
<python基础教程(第二版)>学习笔记 函数(第6章) 创建函数:def function_name(params): block return values 记录函数:def f ...
- 一文搞懂Python函数(匿名函数、嵌套函数、闭包、装饰器)!
Python函数定义.匿名函数.嵌套函数.闭包.装饰器 目录 Python函数定义.匿名函数.嵌套函数.闭包.装饰器 函数核心理解 1. 函数定义 2. 嵌套函数 2.1 作用 2.2 函数变量作用域 ...
随机推荐
- 操作系统学习笔记5 | 用户级线程 && 内核级线程
在上一部分中,我们了解到操作系统实现多进程图像需要组织.切换.考虑进程之间的影响,组织就是用PCB的队列实现,用到了一些简单的数据结构知识.而本部分重点就是进程之间的切换. 参考资料: 课程:哈工大操 ...
- C#基础_类的声明
新建Clerk类. using System; using System.Collections.Generic; using System.Linq; using System.Text; usin ...
- OpenJudge1.5.17 菲波那契数列
17:菲波那契数列 总时间限制: 1000ms 内存限制: 65536kB 描述 菲波那契数列是指这样的数列: 数列的第一个和第二个数都为1,接下来每个数都等于前面2个数之和. 给出一个正整数k,要求 ...
- KFS邮件自动告警-数据比对-数据修复配置方法
一.告警机制 用户可以通过配置告警机制,在比对完成和节点报错时接收到邮件告警. 告警机制共包含3个方面: 1. 告警配置 2. 用户订阅 3. 告警历史 KFS邮箱分两个部分,一个是接收告警信息的邮箱 ...
- 004-GoingDeeperConvolutions2014(googLeNet)
Going Deeper with Convolutions #paper 1. paper-info 1.1 Metadata Author:: [[Christian Szegedy]], [[W ...
- 跟羽夏学 Ghidra ——调试
写在前面 此系列是本人一个字一个字码出来的,包括示例和实验截图.本人非计算机专业,可能对本教程涉及的事物没有了解的足够深入,如有错误,欢迎批评指正. 如有好的建议,欢迎反馈.码字不易,如果本篇文章 ...
- 关于使用kuboard安装其自带的监控应用的注意事项
在安装过程中若想监控kube-controller-manager和kube-scheduler,需要按步骤中的如下说明操作 在这里,所有master节点的这俩文件都需要修改,不用apply,等一分钟 ...
- SQL通用语法和SQL分类
SQL通用语法 1.SQL 语句可以单行或多行书写,以分号结尾 2.可使用空格和缩进来增强语句的可读性 3.MySQL 数据库的SQL语句不区分大小写,关键字建议使用大写 4.3种注释 单行注释: - ...
- C# 传不定参数
1 public class MyClass 2 { 3 public static void UseParams(params int[] list) 4 { 5 for (int i = 0; i ...
- 关于click和onclick的区别
click()和onclick()的区别: 1.onclick是绑定事件,告诉浏览器在鼠标点击时候要做什么 click本身是方法作用是触发onclick事件,只要执行了元素的click()方法,就会触 ...