python装饰器返回值,python装饰器获取被装饰函数的参数
本文已经给大家带来了一些关于Python的知识,主要梳理了高级编程的相关问题,包括返回函数、闭包、decorators、部分函数等。下面就来看看吧,希望对你有帮助。
【相关推荐:Python3视频教程】
00-1010高阶函数不仅可以接受函数作为参数,还可以返回函数作为结果值。我们在操作函数时,如果不需要立即求和,我们会在下面的代码中根据需要进行计算。
例如下面
# -*- coding: utf-8 -*-# python全栈# Author3360a魏#开发时间3360 2022/6/23 22336041 DEF SUM _ FUN _ A(* ARGS)3360
a=0
对于args:中的n
a=a n
这是sum_fun方法,我不需要立即计算我的结果。它不返回求和结果,而是返回求和函数,如下所示
# -*- coding: utf-8 -*-# python全栈# Author3360a魏#开发时间3360 2022/6/23 22336041 DEF SUM _ FUN _ B(* ARGS)3360
def sum_a():
a=0
对于args:中的n
A=a n return a返回sum_a当我们调用sum_fun_b()时,返回的不是求和结果,而是求和函数sum_a,在我们调整sum_fun_b函数时赋给变量。
F1=sum_fun_b(1,2,3,4,5)#此时F被实例化为对象,值print (f1 ()) # 15f2=sum _ fun _ b (1,2,3,4,5) F3=sum _ fun _ b (1 f3)函数sum _ fun _ b . locals . sum _ a At0x 0000016 E1 e EFD 30函数sum _ fun _ b . locals . sum _ a At0x 0000016 E1 e eff 700 print(id(F2),ID此时f=sum_a,那么这里就有一个问题了。参数去哪了?
而且我们可以看到,创建的两个方法互不影响,地址和值都不一样。
函数sum_a定义在函数sum_fun_b中,内部函数sum_a可以引用外部函数sum_fun_b的参数和局部变量,当sum _ fun _ b返回函数sum_a时,对应的参数和变量存储在返回的函数中,这里称为闭包。
00-1010什么是闭包?
先看一段代码。
#定义一个函数def fun_a(num_a):#在函数内部定义另一个函数#并且这个内部函数去一个外部变量。这个函数以及外部函数的变量和参数被称为闭包。
def fun_b(num_b):
Print(内嵌函数fun_b的参数是:%s,外部函数fun_a的参数是:%s% (num_b,num_a))
退货数量a数量b
#这里返回的是闭包的结果。
Return _ b #给fun_a函数赋值,这个10传递给fun_aret=fun_a(10)#注意这里的10实际上是赋给fun_bprint(ret(10))#注意这里的90实际上是赋给fun_bprint(ret(90)) 3360的运行结果
内嵌函数fun_b的参数是:10,外部函数fun_a的参数是33601020,内嵌函数fun_b的参数是336090,外部函数fun_a的参数是336010100。此时,内部函数将外部函数作为全局变量引用时,称为闭包。
这里有三个关闭的条件。
三个条件缺一不可:
1)必须有一个嵌入函数(函数中定义的函数)——,它嵌套在对应的函数之间。
2)嵌入式函数必须引用在封闭范围内定义的变量(在外部函数中)。——内部函数是指外部变量。
3)外部函数必须返回嵌入函数3354,必须返回哪个内部函数。
python# python交互式环境编辑器G
t;>> def counter(start=0):
count = [start]
def incr():
count[0] += 1
return count[0]
return incr
>>> c1 = counter(5)>>> print(c1()) 6>>> print(c1()) 7>>> c2=counter(50) >>> print(c2()) 51>>> print(c2()) >52>>>当一个函数在本地作用域找不到变量申明时会向外层函数寻找,这在函数闭包中很常见但是在本地作用域中使用的变量后,还想对此变量进行更改赋值就会报错
def test():报错信息:count = 1
def add():
print(count)
count += 1
return add
a = test() a()
Traceback (most recent call last): ...... UnboundLocalError: local variable 'count' referenced before assignment如果我在函数内加一行nonlocal count就可解决这个问题
代码
# -*- coding: UTF-8 -*- # def test():nonlocal声明的变量不是局部变量,也不是全局变量,而是外部嵌套函数内的变量。# count不是局部变量,介于全局变量和局部变量之间的一种变量,nonlocal标识
count = 1
def add():
nonlocal count
print(count)
count += 1
return count return add
a = test() a() # 1 a() # 2
如果从另一个角度来看我们给此函数增加了记录函数状态的功能。当然,这也可以通过申明全局变量来实现增加函数状态的功能。当这样会出现以下问题:
1. 每次调用函数时,都得在全局作用域申明变量。别人调用函数时还得查看函数内部代码。使用nonlocal的好处是,在为函数添加状态时不用额外地添加全局变量,因此可以大量地调用此函数并同时记录着多个函数状态,每个函数都是独立、独特的。针对此项功能其实还个一个方法,就是使用类,通过定义__call__ 可实现在一个实例上直接像函数一样调用3. 当函数在多个地方被调用并且同时记录着很多状态时,会造成非常地混乱。
代码如下:
def line_conf(a, b):运行结果为def line(x):
return a * x + b
return line
line1 = line_conf(1, 1) line2 = line_conf(4, 5) print(line1(5))
print(line2(5))
625从这段代码中,函数line与变量a,b构成闭包。在创建闭包的时候,我们通过line_conf的参数a,b说明了这两个变量的取值,这样,我们就确定了函数的最终形式(y = x + 1和y = 4x + 5)。我们只需要变换参数a,b,就可以获得不同的
直线表达函数。由此,我们可以看到,闭包也具有提⾼代码可复⽤性的作⽤。如果没有闭包,我们需要每次创建函数的时候同时说明a,b,x。这样,我们就需要更多的参数传递,也减少了代码的可移植性。
1.闭包似优化了变量,原来需要类对象完成的⼯作,闭包也可以完成但是还没有结束,我们知道,函数内部函数,引用外部函数参数或值,进行内部函数运算执行,并不是完全返回一个函数,也有可能是一个在外部函数的值,我们还需要知道返回的函数不会立刻执行,而是直到调用了函数才会执2.由于闭包引⽤了外部函数的局部变量,则外部函数的局部变量没有及时释放,消耗内存
行。
看代码:
def fun_a():这里创建了一个fun_a函数,外部函数的参数fun_list定义了一个列表,在进行遍历,循环函数fun_b,引用外部变量i 计算返回结果,加入列表,每次循环,都创建了一个新的函数,然后,把创建的3个函数都返回了fun_list = []
for i in range(1, 4):
def fun_b():
return i * i
fun_list.append(fun_b)
return fun_list
f1, f2, f3 = fun_a() print(f1(), f2(), f3())# 结果:9,9,9
但是实际结果并不是我们想要的1,4,9,而是9,9,9,这是为什么呢?
这是因为,返回的函数引用了变量 i ,但不是立刻执行。等到3个函数都返回时,它们所引用的变量i已经变成了3,每一个独立的函数引用的对象是相同的变量,但是返回的值时候,3个函数都返回时,此时值已经完整了运算,并存储,当调用函数,产生值不会达成想要的,返回函数不要引用任何循环变量,或者将来会发生变化的变量,但是如果一定需要呢,如何修改这个函数呢?
我们把这里的i赋值给_就可以解决
def test3():可以再创建一个函数,用该函数的参数绑定循环变量当前的值,无论该循环变量后续如何更改,已绑定到函数参数的值不变,那我们就可以完成下面的代码func_list = []
for i in range(1, 4):
def test4(i_= i):
return i_**2
func_list.append(test4)
return func_list
f1, f2, f3 = test3()print(f1(), f2(), f3())
# -*- coding: UTF-8 -*- # def fun_a():def fun_c(i):
def fun_b():
return i * i
return fun_b
fun_list = []
for i in range(1, 4):
# f(i)立刻被执行,因此i的当前值被传入f()
fun_list.append(fun_c(i))
return fun_list
f1, f2, f3 = fun_a() print(f1(), f2(), f3()) # 1 4 9
3.装饰器 wraps
什么是装饰器?看一段代码:
# -*- coding: utf-8 -*-# python 全栈# author : a wei # 开发时间: 2022/6/24 17:03def eat():由于函数也是一个对象,而且函数对象可以被赋值给变量,所以,通过变量也能调用该函数。也可以将函数赋值变量,做参传入另一个函数。print('吃饭')def test1(func):
def test2():
print('做饭')
func()
print('洗碗')
return test2
eat() # 调用eat函数# 吃饭test1(eat)()# 做饭# 吃饭# 洗碗
1>什么是装饰器
"""装饰器的作用就是为已经存在的对象添加额外的功能装饰器本质上是一个Python函数,它可以让其他函数在不需要做任何代码变动的前提下增加额外功能,装饰器的返回值 也是一个函数对象。
它经常用于有以下场景,比如:插入日志、性能测试、事务处理、缓存、权限校验等场景。装饰器是解决这类问题的绝 佳设计
"""
先看代码:
# -*- coding: utf-8 -*-# python 全栈# author : a wei # 开发时间: 2022/6/24 17:03def test1(func):我们没有直接将eat函数作为参数传入test1中,只是将test1函数以@方式装饰在eat函数上。def test2():
print('做饭')
func()
print('洗碗')
return test2@test1 # 装饰器def eat():
print('吃饭')eat()# 做饭# 吃饭# 洗碗
也就是说,被装饰的函数,函数名作为参数,传入到装饰器函数上,不影响eat函数的功能,再此基础上可以根据业务或者功能增加条件或者信息。
(注意:@在装饰器这里是作为Python语法里面的语法糖写法,用来做修饰。)
但是我们这里就存在一个问题这里引入魔术方法 name 这是属于 python 中的内置类属性,就是它会天生就存在与一个 python 程序中,代表对应程序名称,一般一段程序作为主线运行程序时其内置名称就是 main ,当自己作为模块被调用时就是自己的名字
代码:
print(eat.__name__)# test2这并不是我们想要的!输出应该是" eat"。这里的函数被test2替代了。它重写了我们函数的名字和注释文档,那怎么阻止变化呢,Python提供functools模块里面的wraps函数解决了问题
代码:
-*- coding: utf-8 -*-我们在装饰器函数内,作用eat的test2函数上也增加了一个装饰器wraps还是带参数的。from functools import wrapsdef test1(func):
@wraps(func)
def test2():
print('做饭')
func()
print('洗碗')
return test2@test1 # 装饰器def eat():
print('吃饭')eat()# 做饭# 吃饭# 洗碗print(eat.__name__)# eat
这个装饰器的功能就是不改变使用装饰器原有函数的结构。
我们熟悉了操作,拿来熟悉一下具体的功能实现,我们可以写一个打印日志的功能
# -*- coding: utf-8 -*-# python 全栈# author : a wei # 开发时间: 2022/6/24 17:42import timefrom functools import wrapsdef logger(func):@wraps(func)
def write_log():
print('[info]--时间:%s' % time.strftime('%Y-%m-%d %H:%M:%S'))
func()
return write_log@loggerdef work():
print('我在工作')work()# [info]--时间:2022-06-24 17:52:11# 我在工作print(work.__name__)#work
2>带参装饰器
我们也看到装饰器wraps也是带参数的,那我们是不是也可以定义带参数的装饰器呢,我们可以使用一个函数来包裹装饰器,调入这个参数。
# -*- coding: utf-8 -*-# python 全栈# author : a wei # 开发时间: 2022/6/24 17:42import timefrom functools import wrapsdef logs(func):@wraps(func)
def write_log(*args, **kwargs):
print('[info]--时间:%s' % time.strftime('%Y-%m-%d %H:%M:%S'))
func(*args, **kwargs)
return write_log@logsdef work():
print('我在工作')@logsdef work2(name1, name2):
print('%s和%s在工作' % (name1, name2))work2('张三', '李四')# [info]--时间:2022-06-24 18:04:04# 张三和李四在工作
3>函数做装饰器
把日志写入文件
# -*- coding: utf-8 -*-# python 全栈# author : a wei# 开发时间: 2022/6/20 0:06import timefrom functools import wrapsdef logger(file):终端输出:def logs(fun):
@wraps(fun)
def write_log(*args, **kwargs):
log = '[info] 时间是:%s' % time.strftime('%Y-%m-%d %H:%M:%S')
print(log)
with open(file, 'a+') as f:
f.write(log)
fun(*args, **kwargs)
return write_log return logs@logger('work.log') # 使用装饰器来给 work函数增加记录日志的功能def work(name, name2): # 1.当前 work可能有多个参数 2.自定义日志文件的名字和位置,记录日志级别
print(f'{name}和{name2}在工作')work('张三', '李四')
这里生成里work.log日志文件
里面记录日志
这里我们将带参数的带入进去根据代码流程执行生成了文件并将文件打印进去现在我们有了能用于正式环境的logs装饰器,但当我们的应用的某些部分还比较脆弱时,异常也许是需要更紧急关注的事情。
比方说有时你只想打日志到一个文件。而有时你想把引起你注意的问题发送到一个email,同时也保留
日志,留个记录。
这是一个使用继承的场景,但目前为止我们只看到过用来构建装饰器的函数。
4>类做装饰器
# -*- coding: utf-8 -*-# python 全栈# author : a wei# 开发时间: 2022/6/20 0:06import timefrom functools import wraps# 不使用函数做装饰器,使用类做装饰器class Logs(object):这个实现有一个优势,在于比嵌套函数的方式更加整洁,而且包裹一个函数还是使用跟以前一样的语法def __init__(self, log_file='out.log', level='info'):
# 初始化一个默认文件和默认日志级别
self.log_file = log_file
self.level = level def __call__(self, fun): # 定义装饰器,需要一个接受函数
@wraps(fun)
def write_log(name, name2):
log = '[%s] 时间是:%s' % (self.level, time.strftime('%Y-%m-%d %H:%M:%S'))
print(log)
with open(self.log_file, 'a+') as f:
f.write(log)
fun(name, name2)
return write_log@Logs() # 使用装饰器来给 work函数增加记录日志的功能def work(name, name2): # 1.当前 work可能有多个参数 2.自定义日志文件的名字和位置,记录日志级别
print(f'{name}和{name2}在工作')work('张三', '李四') # 调用work函数
4.偏函数 partial
Python的 functools 模块提供了很多有用的功能,其中一个就是偏函(Partial function)。要注意,这里的偏函数和数学意义上的偏函数不一样。在介绍函数参数的时候,我们讲到,通过设定参数的默认值,可以降低函数调用的难度。而偏函数也可以做到这一点。
例如:int() 函数可以把字符串转换为整数,当仅传入字符串时, int() 函数默认按十进制转换
>>> int('123') 123但 int() 函数还提供额外的 base 参数,默认值为 10 。如果传入 base 参数,就可以做进制的转换
>>> int('12345', base=8) 5349 >>> int('12345', 16) 74565如果要转换大量的二进制字符串,每次都传入 int(x, base=2) 非常麻烦,于是,我们想到,可以定义一个int2() 的函数,默认把 base=2 传进去:
代码:
# 定一个转换义函数 >>> def int_1(num, base=2):把一个函数的某些参数给固定住(也就是设置默认值),返回一个新的函数,调用这个新函数会更简单return int(num, base)
>>> int_1('1000000') 64>>> int_1('1010101') 85
继续优化,functools.partial 就是帮助我们创建一个偏函数的,不需要我们自己定义 int_1() ,可以直接使用下面的代码创 建一个新的函数 int_1
# 导入 >>> import functools理清了 functools.partial 的作用就是,把一个函数的某些参数给固定住(也就是设置默认值),返回一个新的函数,调用这个新函数会更简单。# 偏函数处理 >>> int_2 = functools.partial(int, base=2) >>> int_2('1000000') 64>>> int_2('1010101') 85
注意到上面的新的 int_2 函数,仅仅是把 base 参数重新设定默认值为 2 ,但也可以在函数调用时传入其他值实际上固定了int()函数的关键字参数 base
int2('10010')相当于是:
kw = { base: 2 } int('10010', **kw)当函数的参数个数太多,需要简化时,使用 functools.partial 可以创建一个新的函数,这个新函数可以固定住原函数的部分参数,从而在调用时更简单
【相关推荐:Python3视频教程 】以上就是一文掌握Python返回函数、闭包、装饰器、偏函数的详细内容,更多请关注盛行IT软件开发工作室其它相关文章!
郑重声明:本文由网友发布,不代表盛行IT的观点,版权归原作者所有,仅为传播更多信息之目的,如有侵权请联系,我们将第一时间修改或删除,多谢。