python3 闭包装饰器_python 闭包 装饰器

闭包

1. 什么是闭包

在 python 中创建一个闭包一般有3个要求:

(1)闭包函数必须有内嵌函数

(2)内嵌函数必须要引用外层函数的变量

(3)闭包函数返回内嵌函数的地址(函数名称)

作用:可以在不修改目标源码的前提下,加功能

注意:闭包函数中的变量的生命周期得到延长

2. 创建一个闭包函数

def funcOut():

name = 'Jery'

def funcIn():

# format的简写方式

print(f"姓名 = {name}")

return funcIn

f = funcOut()

f()

运行结果:

姓名 = Jery

3. 判断是否为闭包函数

闭包函数相对与普通函数会多出一个__closure__的属性,里面定义了一个元组用于存放所有的cell对象,每个cell对象保存了这个闭包中所有的外部变量。

funcIn.__closure__:返回None,则不是闭包函数

如:

def funcOut():

name = 'Jery'

def funcIn():

print(f"姓名 = {name}")

print(funcIn.__closure__)

print(funcIn.__closure__[0].cell_contents) # 第一个外部变量

return funcIn

f = funcOut()

f()

运行结果

(,)

Jery

姓名 = Jery

装饰器

1. 本质及作用

装饰器的本质

闭包函数

装饰器的作用:

在不修改原函数及其调用方式的情况下对原函数功能进行扩展

2. 装饰器的使用

需求:为现有功能fun1增加日志功能

传统方案解决 —— 使用闭包

def writeLog(fn):

print("记录日志")

print('访问方法:'+fn.__name__)

def funcOut(func):

def funcIn():

writeLog(func)

func()

return funcIn

def fun1():

print("使用功能1")

def fun2():

print("使用功能2")

fun1 = funcOut(fun1)

# 装饰器(闭包)

fun1()

运行结果:

记录日志

访问方法:fun1

使用功能1

使用装饰器(语法糖)解决

def writeLog(fn):

print("记录日志")

print('访问方法:'+fn.__name__)

def funcOut(func):

def funcIn():

writeLog(func)

func()

return funcIn

@funcOut

def fun1():

print("使用功能1")

@funcOut

def fun2():

print("使用功能2")

fun1()

fun2()

运行结果:

记录日志

访问方法:fun1

使用功能1

记录日志

访问方法:fun2

使用功能2

3. 多个装饰器的使用

如:

def war1(func):

print("war 1")

def inner(*args, **kwargs):

print("======war1 start=====")

func(*args, **kwargs) # inner

print("======war1 end=====")

return inner

def war2(func):

print("war2")

def inner(*args, **kwargs):

print("======war2 start=====")

func(*args, **kwargs)

print("======war2 end=====")

return inner

@war1

@war2

def f():

print("****self****")

f()

运行结果:

war2

war1

======war1 start=====

======war2 start=====

****self****

======war2 end=====

======war1 end=====

解释:

(1)

@war1

@war2 之后相当于 --> f = war1(war2(f))

其中war2(f)是一个函数,作为实参传递

war2(f):

print("======war2 start=====")

print("****self****")

print("======war2 end=====")

war1(war2(f)):

print("======war1 start=====")

war2(f)

print("======war1 end=====")

(2)

f() 相当于执行 --> war1(war2(f))()

4. 对有参数的函数进行装饰

def funcOut(fn):

print("funcOut")

def funcIn(aa, bb):

print("funcIn1")

fn(aa,bb)

print("funcIn2")

return funcIn

@funcOut

def test(a, b):

print("a=%d,b=%d" % (a, b))

# 装饰器装饰之后,这不是直接调用test方法,而是调用func_in方法

test(1,2)

结果

funcOut

funcIn1

a=1,b=2

funcIn2

5. 通用装饰器的使用

一个装饰器可以装饰多个不同参数、不同返回值的函数

如:

def funcOut(fn):

# 需要有参数,*args,**kwargs

def funcIn(*args,**kwargs):

print("记录日志")

print('访问方法:'+fn.__name__)

# 需要有参数,*args,**kwargs

# 需要有返回值

return fn(*args,**kwargs)

return funcIn

# 待装饰函数1:无参数,无返回值

@funcOut

def test1():

print("test1")

test1()

print("---------")

# 待装饰函数2:无参数,有返回值

@funcOut

def test2():

return "Hello"

print(test2())

print("---------")

# 待装饰函数3:有参数,无返回值

@funcOut

def test3(a):

print('a=%d'%a)

test3(1)

print("---------")

# 待装饰函数3:有键值对参数,无返回值

@funcOut

def test4(**kwargs):

print(kwargs)

test4(a=1)

结果

记录日志

访问方法:test1

test1---------记录日志

访问方法:test2

Hello---------记录日志

访问方法:test3

a=1

---------记录日志

访问方法:test3

{'a': 1}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
闭包装饰器是一种特殊的装饰器,它使用闭包的概念来实现闭包是指一个函数可以访问并操作其外部函数中定义的变量。在Python中,闭包装饰器可以用于给函数添加额外的功能,同时保持函数的原始定义不变。 引用中的示例展示了装饰器传参的形式。在这个例子中,outer函数是一个装饰器,它将inner函数作为子函数返回,并在inner函数中添加了额外的功能。通过使用@outer装饰器语法,我们可以在add函数上应用outer装饰器,从而在调用add函数时执行装饰器中的代码。 引用中的示例展示了多层装饰器使用。在这个例子中,outer1和outer2函数分别是两个装饰器,他们都返回一个inner函数。通过使用@outer1和@outer2装饰器语法,我们可以在outers函数上应用这两个装饰器,并在调用outers函数时按照装饰器的定义顺序执行相关的代码。 引用提供了关于Python闭包装饰器使用方法的总结。这篇文章通过示例代码详细介绍了闭包装饰器使用,对于学习和工作有一定的参考价值。 引用中的示例展示了装饰器的形式。在这个例子中,outer函数是一个装饰器,它将inner函数作为子函数返回,并在inner函数中添加了额外的功能。通过使用@outer装饰器语法,我们可以在add函数上应用outer装饰器,从而在调用add函数时执行装饰器中的代码。 综上所述,Python闭包装饰器是一种利用闭包概念实现的特殊装饰器,可以用于给函数添加额外的功能。这种装饰器可以通过装饰器传参的形式、多层装饰器的形式或普通的装饰器形式来实现。<span class="em">1</span><span class="em">2</span><span class="em">3</span><span class="em">4</span>

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值