Python闭包与装饰器

闭包

1.函数引用

def test1():
    print("---in test1 func---")
    
#调用函数
test1()

#引用函数
ret=test1

print(id(ret))
print(id(test1))

#通过引用调用函数
ret()

运行结果:

---in test1 func---
123456789000
123456789000
---in test1 func---

2.什么是闭包

#定义一个函数
def test(number):
    
    #在函数内部在定义一个函数,并且这个函数用到了外边函数的变量,那么将这个函数称为闭包
    def test_in(number_in):
        print("in test_in 函数,number_in is %d"%number_in)
        return number+number_in
    return test_in

#给test函数赋值,这个10就是给参数number    
ret=test(10)

#注意这里的100其实给参数number_in
print(ret(100))

#注意这里的200其实给参数number_in
print(ret(200))

运行结果:

in test_in函数,number_in is 100
110
in test_in函数,number_in is 200
210

3.重讲闭包

内部函数对外部函数作用域里变量的引用(非全局变量)则称内部函数为闭包。

#test2.py
def counter(start=0):
    count=[start]
    def incr():
        count[0]+=1
        return count[0]
    return incr

应用test2.py

import test2
c1=test2.counter(10)
print(c1())

结果:11
print(c1())
结果:12

nonlocal访问外部函数的局部变量

def counter(start=0):
    def incr():
        nonlocal start
        start+=1
        return start
    return incr

c1=counter(5)
print(c1())
print(c1())

c2=counter(50)
print(c2())
print(c2())

print(c1())
print(c1())

print(c2())
print(c2())

4.实例

def line_conf(a,b):
    def line(x):
        return a*x+b
    return line

line1=line_conf(1,2)
line2=line_conf(3,4)
print(line1(5))
print(line2(6))

在这个例子中,函数line与变量a,b构成闭包,在创建闭包的时候,我们通过line_conf的参数a,b说明了这两个变量的取值,这样,我们就确定了函数的最终形式(y=x+2和y=3x+4)。我们只需要变化参数a,b,就可以获得不同的直线表达函数,因此,我们可以看到,闭包也具有提高代码可复用性的作用。

如果没有闭包,我们需要每次创建直线函数的时候同时说明a,b,x。这样,我们就需要传递更多的参数

装饰器

1.首先理解一段代码

def foo():
    print('foo')
    
foo  #表示是函数
foo()  #表示执行foo函数


def foo():
    print('foo')
    
foo=lambda x: x+1
foo()  #执行下面的lambda表达式,而不是原来的foo函数

2.需求

某公司由N个业务部门和一个基础平台部门构成,基础平台负责提供底层的功能,如:数据库操作、redis调用、监控API等功能。业务部门使用基础功能时,只需调用基础平台提供的功能即可。如:

"""基础平台提供的功能如下"""
def f1():
    print('f1')
    
def f2():
    print('f2')
    
def f3():
    print('f3')

def f4():
    print('f4')
    
    
    
"""业务部门a调用基础平台提供的功能"""
f1()
f2()
f3()
f4()


"""业务部门b调用基础平台提供的功能"""
f1()
f2()
f3()
f4()

基础平台提供的功能可以被任何人使用。现在需要对基础平台的所有功能进行重构,为平台提供所有功能添加验证机制,既要在执行功能前进行验证。

程序员A的做法:

"""基础平台提供的功能"""

def f1():
    #验证1
    #验证2
    #验证3
    print('f1')
 
def f2():
    #验证1
    #验证2
    #验证3
    print('f2')
    
   
def f3():
    #验证1
    #验证2
    #验证3
    print('f3')
    
   
def f4():
    #验证1
    #验证2
    #验证3
    print('f4')
   

"""业务部门代码不变"""
"""A,B分别调用"""
f1()
f2()
f3()
f4()

程序员B的做法:

只对基础平台的代码进行重构,其他业务部门无需做任何修改

"""基础平台提供的功能如下"""

def check_login():
    #验证1
    #验证2
    #验证3
    pass

def f1():
    check_login()
    print('f1')
    
def f2():
    check_login()
    print('f2')
   
def f3():
    check_login()
    print('f3')
   
def f4():
    check_login()
    print('f4')
    
    

程序员C的做法:

写代码遵循开放封闭原则,虽然在这个原则是用面向对象开发,但是也适用于函数式编程,简单来说,它规定已经实现的代码不允许被修改,但可以被扩展即:

  • 封闭:对实现的功能代码块
  • 开放:对扩展开发

若实现上述需求,则不允许在f1、f2、f3、f4内部进行修改代码,实现方案:

def w1(func):
    def inner():
        #验证1
        #验证2
        #验证3
        func()
    return inner

@w1
def f1():
    print('f1')
    
def f2():
    print('f2')
    
def f3():
    print('f3')
   
def f4():
    print('f4')
    
    

执行过程

python解释器将代码从上至下进行解释:

  1. def w1(func):将函数w1加载到内存;
  2. @w1
f1=w1(f1)

表面上解释器仅仅会解释这两句代码,因为函数在没有被调用之前其内部代码不会被执行。

从表面上看解释器会执行这两句,但是@w1这一句在python中是一种语法糖(@函数名)

@w1内部会执行以下操作

执行w1函数

执行w1函数,并将@w1下面的函数作为w1函数的参数,即:@w1等价于w1(f1)。所以,内部就会去执行:

def inner():
    #验证1
    #验证2
    #验证3
    f1()     #func是参数,此时func等于f1
    
return inner  #返回的inner,inner代表的是函数,非执行函数,
w1的返回值

将执行完的2w1函数返回值赋值为@w1下面的函数f1,即将w1的返回值再重新赋值为w1,即:

新
f1=def inner():
    #验证1
    #验证2
    #验证3
    原f1()
return inner

所以,以后业务部门想要执行f1函数时,就会执行新f1函数,在新f1函数内部先执行验证,在执行原来的f1函数,然后将原来f1函数的返回值返给了业务调用者。

如此一来,即执行了验证功能,又执行了原来f1函数的内容,并将原f1函数返回给业务调用者。

3.再议装饰器

#定义函数:完成数据包裹
def makeBold(fn):
    def wrapped():
        return "<b>"+fn()+"</b>"
    return wrapped

#定义函数:完成数据包裹
def makeItalic(fn):
    def wrapped():
        return "<i>"+fn()+"</i>"
    return wrapped

@makeBold
def test1():
    return "hello world-1"

@makeItalic
def test2():
    return "hello world-2"

@makeBold
@makeItalic
def test3():
    return "hello world-3"

print(test1())
print(test2())
print(test3())

运行结果:

<b>hello world-1</b>
<i>hello world-2</i>
<b><i>hello world-3</i></b>

4.装饰器(decorator)功能

  1. 引入日志
  2. 函数执行时间统计
  3. 执行函数前预备处理
  4. 执行函数后清理功能
  5. 权限校验等场景
  6. 缓存

5.示例

例1.无参数的函数
from time import ctime,sleep

def timefun(func):
    def wrappedfunc():
        print("%s called at %s"%(func.__name__,ctime()))
        func()
    return wrappedfunc

@timefun
def foo():
    print("I am foo")
    
foo()
sleep(2)
foo()

上面代码装饰器执行过程可表述为:

foo=timefun(foo)
#foo先作为参数赋值给func后,foo接收指向timefun返回的wrappedfunc
foo()
#调用foo(),即等价于调用wrappedfunc()
#内部函数wrappedfunc被引用,所以外部函数的func变量(自由变量)并没有释放
#func里保存的是原foo函数对象

代码执行结果

执行结果.png
例2.被装饰的函数有参数
from time import ctime,sleep

def timefun(func):
    def wrappedfunc(a,b):
        print("%s called at %s"%(func.__name__,ctime()))
        print(a,b)
        func(a,b)
    return wrappedfunc

@timefun
def foo(a,b):
    print(a+b)
    
    
foo(2,8)
sleep(2)
foo(4,6)

执行结果

执行结果.png
例3.被装饰的函数有不定长参数
from time import ctime,sleep

def timefun(func):
    def wrappedfunc(*args,**kwargs):
        print("%s called at %s"%(func.__name__,ctime()))
        func(*args,**kwargs)
    return wrappedfunc

@timefun
def foo(a,b,c):
    print(a+b+c)
    
foo(1,3,5)
sleep(2)
foo(2,4,6)

执行结果

执行结果.png
例4,装饰器中的return
from time import ctime,sleep

def timefun(func):
    def wrappedfunc():
        print("%s called at %s"%(func.__name__,ctime()))
        func()
    return wrappedfunc

@timefun
def foo():
    print('I am foo')
    
@timefun
def getInfo():
    return '---hello---'

foo()
sleep(2)
foo()


print(getInfo())

运行结果

执行结果.png

如果修改装饰器为return func,则运行结果为:

执行结果.png

或者参考如下代码

from time import ctime,sleep

def timefun(func):
    def wrappedfunc():
        print("%s called at %s"%(func.__name__,ctime()))
        ret=func()  #保存返回来的hello
        return ret   #把hello返回给调用函数
    return wrappedfunc

@timefun
def foo():
    print('I am foo')
    
@timefun
def getInfo():
    return '---hello---'

foo()
sleep(2)
foo()


print(getInfo())
例5.装饰器带参数,在原有装饰器的基础上,设置外部变量
from time import ctime, sleep

def timefun_arg(pre="hello"):
    def timefun(func):
        def wrappedfunc():
            print("%s called at %s %s"%(func.__name__, ctime(), pre))
            return func()
        return wrappedfunc
    return timefun

@timefun_arg("itcast")
def foo():
    print("I am foo")

@timefun_arg("python")
def too():
    print("I am too")

foo()
sleep(2)
foo()

too()
sleep(2)
too()

可以理解为:

foo()==timefun_arg("itcast")(foo)()
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,717评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,501评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,311评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,417评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,500评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,538评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,557评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,310评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,759评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,065评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,233评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,909评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,548评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,172评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,420评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,103评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,098评论 2 352

推荐阅读更多精彩内容

  • 运行结果如下: 闭包的定义:在函数内部再定义一个函数,并且这个函数用到了外边函数的变量,那么将这个函数以及用到的一...
    魔法高校的劣等生阅读 517评论 0 0
  • # 本次讲述的知识点也是非常重要的(严肃脸) # 先有的闭包,之后才生成了装饰器,同样的也是非常简单的东西。 # ...
    米兰的小铁匠阅读 463评论 0 4
  • python万物皆对象! 闭包 闭包:两个函数的嵌套,外部函数返回内部函数的引用,外部函数一定有参数 def 外部...
    风舞柏杨阅读 777评论 2 0
  • 1.1==,is的使用 ·is是比较两个引用是否指向了同一个对象(引用比较)。 ·==是比较两个对象是否相等。 1...
    TENG书阅读 728评论 0 0
  • ​前几天在医院排队的时候,想发一个关于我妈妈的段子,主要想吐槽一下她后妈的体质。于是编辑了如下一段: 好多人问我,...
    芳言芳语阅读 565评论 2 2