闭包函数和装饰器

闭包函数

定义:闭是封闭(函数内部的函数),闭包指的是内部函数包含对外部作用域而非全局作用域的引用。

函数对象:可以将定义在函数内的函数返回到全局使用,从而打破函数的层级限制。

def outter():
    x = 1

    def inner():
        print(x)
    return inner

f = outter()

def f2():
    x = 2
    f()

f2()


# 1

为函数传参的两种方式

1.使用参数的形式

def func(x):
    print(x)

func(1)
func(1)
func(1)

'''
1
1
1
'''

2.包给函数

ef outter(x):
    x = 1

    def inner():
        print(x)
    return inner

f = outter(1)
f()
f()
f()
'''
1
1
1
'''

闭包函数的应用

闭包的意义:返回的函数对象,不仅仅是一个函数对象,在该函数外还包裹了一层作用域,这使得,该函数无论在何处调用,优先使用自己外层包裹的作用域。

# 普通的传参的方式   每用一次需要传一次参数,显得极其麻烦

import requests


def get(url):
    response = requests.get(url)
    print(f"done: {url}")
    
get('https://www.baidu.com')
get('https://www.baidu.com')
get('https://www.baidu.com')


get('https://www.cnblogs.com/linhaifeng')
get('https://www.cnblogs.com/linhaifeng')
get('https://www.cnblogs.com/linhaifeng')
# 把传的参数包给函数后,把内层函数对象返回到外层,在外层直接调用该函数对象即可,不用频繁的传值了。

import requests
def outter(url):
    def get():
        response = requests.get(url)
        print(f"done: {url}")
    return get

baidu=outter('https://www.baidu.com')
python = outter('https://www.python.org')

baidu()
baidu()

python()
python()

装饰器

1. 什么是装饰器
器:指的是具备某一功能的工具
装饰:指的是为被装饰器对象添加新功能

装饰器就是用来为被装饰器对象添加新功能的工具
注意:装饰器本身可以是任意可调用对象,被装饰器的对象也可以是任意可调用对象

2. 为何要用装饰器
开放封闭原则:封闭指的是对修改封闭,对扩展开放
装饰器的实现必须遵循两大原则:
1). 不修改被装饰对象的源代码
2). 不修改被装饰器对象的调用方式
装饰器的目标:就是在遵循1和2原则的前提下为被装饰对象添加上新功能

3.装饰器的应用(装饰器其实也是闭包函数的一种应用场景)

上面的这张图中outter()的功能就是装饰器,它装饰了index函数,他在没有改变被调用对象的源代码和调用方式的前提下,实现了统计函数运行时间的功能,感官体验好像还是index()的调用。

装饰器升级版:装饰器同时装饰两个函数。下图中outter()函数是装饰器,index函数和home函数是被装饰对象。index()函数(其实是wrapper函数)不需要传参,而home()函数(其实是wrapper函数)需要传参,那么要想外部调用home()(其实是wrapper函数)就需要给定义阶段的wrapper()传形参,比如wrapper(name),然后调用wrapper()阶段才能传实参,但一个装饰器在同时需要兼顾两个函数,一个需要传值,另一个不需要传值,同时调用,就会报错。那么我们该如何解决呢????这时我们就需要对装饰器outter()内部函数wrapper()传入两个参数了叫做*args和**kwargs,众所周知这两个参数的作用(在形参中,这两个参数可以接收实参传来的多余位置参数和关键字参数;在实参中,*args这个参数可以将其后面的元组打散成位置实参给位置形参传值,**kwargs这个参数可以将其后面的字典形式的关键字参数打散,传给形参中的关键字参数),所以上图中index不传参,*args和**kwargs没有接收到多余的参数,形成了一个空的()和空的{},在下面调用func()的时候,func括号中(即index括号中)的*args将空元组打散,**kwarg将空字典打散,就没有什么参数传给index函数了。而home()传了位置实参“zhang”,wrapper()形参中的*args默认接收了形参中多余的位置实参‘zhang’,将‘zhang’存元组的形式,而此时**kwargs还是一个空字典,到下面遇到func(*args,**kwargs),*args将(‘zhang’)打散成字符串原封不动的传给了*args,**kwargs的一个空字典被打散没有了。这样两个函数都被装饰上了,添加上了新的功能,也并没有改变原函数的代码,也没改变他们的调用方式。

注意:现在以home函数为例,如果home函数内部有个返回值(return 234),那么加上装饰器outter(),接收outter(home)的返回值,print(res=outter(home)) 返回值是None,因为这里的home()就是wrapper()。

装饰器的语法糖

在被装饰函数上面一行加上@+装饰器,即,例如:

@timmer

def func()

表示装饰func()函数,意思是func=timmer(func),func是最原始的那个func的内存地址。

 运行原理:
 python解释器一旦运行到@装饰器的名字,就会调用装饰器,然后将被装饰函数的内存地址当作参数
 传给装饰器,最后将装饰器调用的结果赋值给原函数名

         

 装饰器模板

import time
def outter(func):
def wrapper(*args,**kwargs):
#在调用函数前加功能
res=func(*args,**kwargs) #调用被装饰的\也就是最原始的那个函数
 在调用函数后加功能
return res
return wrapper

@outter #index=outter(index) #index=wrapper
def index():
print('welcome to index page')
time.sleep(3)

index()

叠加多个装饰器

import time

def timmer(func):  #func=wrapper2
    def wrapper1(*args,**kwargs):
        start=time.time()
        res=func(*args,**kwargs)  #res=wrapper2(*args,**kwargs)
        stop=time.time()
        print('run time is %s' %(stop - start))
        return res
    return wrapper1

def auth(func): #func=最原始的那个index的内存地址
    def wrapper2(*args,**kwargs):
        inp_user = input('please input your username: ').strip()
        inp_pwd = input('please input your password: ').strip()
        if inp_user == 'zhang' and inp_pwd == '123':
            print('login successfull')
            res=func(*args,**kwargs) # 调用最原始的那个/也就是被装饰的那个函数
            return res
        else:
            print('username or password error')
    return wrapper2

# 解释@语法的时候是自下而上运行
# 而执行装饰器内的那个wrapper函数时的是自上而下
@timmer # index=timmer(wrapper2) #index=wrapper1
@auth # index=auth(最原始的那个index的内存地址) #index=wrapper2
def index():
    print('welcome to index page')
    time.sleep(3)

index() #wrapper1() # 首先调用wrapper1函数
'''
有参装饰器由于需要外部传值,而原无参装饰器外层和内层括号内参数又不能动,所以只能在外层再定义一层函数,
利用必报函数的思想,为内部传值,而且不限定传值的个数
,所以装饰器最多也只有三层
'''
# 有参装饰器的模板
def outter1(x,y,z):
    def outter2(func):
        def wrapper(*args,**kwargs):
            res=func(*args,**kwargs)
            return res
        return wrapper
    return outter2

# 无参装饰器的模板
def outter(func):
    def wrapper(*args,**kwargs):
        res=func(*args,**kwargs)
        return res
    return wrapper

有参装饰器应用实例:

import time

current_user={'username':None}

# 补充:所有的数据类型的值自带布尔值,可以直接当作条件去用,只需要记住布尔值为假的那一些值即可(0,空,None)


def login(engine='file'): #engine='mysql'
    def auth(func): #func=最原始那个index的内存地址
        def wrapper(*args,**kwargs):
            if current_user['username']:
                print('已经登录过了,无需再次登陆')
                res=func(*args,**kwargs)
                return res

            if engine == 'file':
                inp_user = input('please input your username: ').strip()
                inp_pwd = input('please input your password: ').strip()
                if inp_user == 'egon' and inp_pwd == '123':
                    print('login successfull')
                    current_user['username']=inp_user # 在登陆成功之后立刻记录登录状态
                    res=func(*args,**kwargs) # res=最原始那个index的内存地址(*args,**kwargs)
                    return res
                else:
                    print('username or password error')
            elif engine == 'mysql':
                print('基于mysql的认证机制')
            elif engine == 'ldap':
                print('基于ldap的认证机制')
            else:
                print('无法识别的认证源')
        return wrapper
    return auth

@login('file')  #@auth # index=auth(最原始那个index的内存地址) #index=wrapper
def index():
    print('welcome to index page')
    time.sleep(3)

@login('file')
def home(name):
    print('welcome %s to home page' %name)
    time.sleep(2)
    return 123


index() #wrapper()
res=home('egon')
print(res)
View Code

伪装真正的装饰器

'''
wraps装饰器应该加到装饰器最内层的函数上,wraps是别人已经写好的装饰器,加在装饰器内层函数上,括号内将原函数名写上,就可以将原函数中的全部内置方法赋值给
最内层对应函数的内置方法上,以达到真正的装饰效果,即客户调用index.__name__或help(index),查到wrapper的内存地址。
''' from functools import wraps import time def deco(func): @wraps(func) def wrapper(*args, **kwargs): res = func(*args, **kwargs) return res # wrapper.__name__=func.__name__ # wrapper.__doc__=func.__doc__ return wrapper @deco #index=deco(index) #index=wrapper函数的内存地址 def index(): """ index 功能 """ print('welcome to index page') time.sleep(3) # print(index.__name__) # 仍然是wrapper,如果客户查看index的函数名,发现是wrapper就露馅了, # print(help(index)) #index.__doc__ __doc__查看函数内部注释信息 # index() # wrapper print(index.__name__) print(index.__doc__)
原文地址:https://www.cnblogs.com/zhangchaocoming/p/11182922.html