python闭包,装饰器,装饰器工厂,类装饰器

1. 闭包

在了解装饰器之前先要知道什么是闭包
闭包概念:内部函数及其使用的环境变量构成的整体叫做闭包
还有对函数进行深入的理解,当你定义一个函数func() 时候,你的函数名只是一个指向代码空间的对象 —> 可以指向别的代码空间
所以,fun变量,只有当加上()时候,才会去执行我们的函数
环境变量: 在函数a内部定义一个函数b,函数a的所有变量都叫做函数b的环境变量。

定义一个函数
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

来看一个闭包的实际例子:

	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))

line1 line2指向的 内部函数是完全独立的数据空间

为什么我们要运行闭包

这个例子中,函数line与变量a,b构成闭包。在创建闭包的时候,我们通过line_conf的参数a,b说明了这两个变量的取值,这样,我们就确定了函数的最终形式(y = x + 1和y = 4x + 5)。
我们只需要变换参数a,b,就可以获得不同的直线表达函数。由此,我们可以看到,闭包也具有提高代码可复用性的作用。
如果没有闭包,我们需要每次创建直线函数的时候同时说明a,b,x。这样,我们就需要更多的参数传递,也减少了代码的可移植性。
在装饰器中,如果我们不用闭包,每当一个装饰器装饰的时候,就会直接运行该函数,而不是等我们自己运行fun()。

注意点:
由于闭包引用了外部函数的局部变量,则外部函数的局部变量没有及时释放,消耗内存

关于修改外部函数的变量(环境变量)

Python3中专门用来修改环境变量的关键字 nonlocal 类似于global一样的用法,先用nonlocal 声明一下,接下来就可以对变量进行更改

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

python2中没有nonlocal这个关键字 — 只能通过([])修改值。这种方式在python3中也同样可以

def counter(start=0):
    count=[start]
    def incr():
        count[0] += 1
        return count[0]
    return incr

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

2. 装饰器

  • 装饰器函数:@装饰器的名称
  • 装饰器函数的特点 参数只有一个 就是被装饰的函数的引用
  • 在内部函数中调用被装饰的函数功能 — 在调用该函数之前和之后都可以加上拓展的功能
  • 返回内部函数的引用

简单实例:

def yanzheng(func):
   def inner():
        print("正在验证")
        # 执行被装饰的函数的 功能
        func()
        print("验证完成")
    return inner

@yanzheng
def f1():
    print("f1")

"""等价于下面的代码
def f1():
    print("f1")
f1 = yanzheng(f1)
"""

@yanzheng
def f2():
    print("f2")

f1()
f2()

重点理解:
f1 = yanzheng(f1) 等价于 @yanzheng  大牛们把枯燥的规格,转变成了优雅的格式,本质是一样的只是换了一下格式
用装饰器来实现一个小功能:统计程序运行时间

import time


def get_time(func):
    def inner():
        begin = time.time()
        # 执行被装饰的函数的 功能
        func()
        end = time.time()
        print("花费了%fs" % (end-begin))

    return inner

@get_time
def fun():
    for i in range(10):
        time.sleep(0.5) 

fun()

2.1 多个装饰器修饰一个函数:

def makeBold(fn):
    def wrapped():
        return "<b>" + fn() + "</b>"
    return wrapped

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

# f1 = makeBold(f1 = makeItalic(f1))
@makeBold
@makeItalic
def f1():
    return "python"

print(f1())

结果:<b><i>hello world-3</i></b>

2.2 装饰有任意参数的函数:

import time

def get_time(func):
    def inner(*args, **kwargs):
        # args接收所有的位置参数 放到args中
        # kwargs接收所有的命名参数 放到kwargs
        begin = time.time()
        # 执行被装饰的函数的 功能  *args **kwargs代表解包、拆包
        ret = func(*args, **kwargs)
        end = time.time()
        print("花费了%fs" % (end-begin))
        return ret

    return inner

# func0 = get_time(fun0)
@get_time
def fun0(number):
    print("in fun0 %d" % number)
    return 100

@get_time
def fun1():
    print("in fun1")
    for i in range(8):
        time.sleep(0.3)
    return 99

print(fun0(9))
print(fun1())  # inner()

2.3 装饰有函数返回值的函数:

import time

def get_time(func):
    def inner(*args, **kwargs):
        # args接收所有的位置参数 放到args中
        # kwargs接收所有的命名参数 放到kwargs
        begin = time.time()
        # 执行被装饰的函数的 功能  *args **kwargs代表解包、拆包
        ret = func(*args, **kwargs)
        end = time.time()
        print("花费了%fs" % (end-begin))
        return ret

    return inner

# func0 = get_time(fun0)
@get_time
def fun0(number):
    print("in fun0 %d" % number)
    return 100

@get_time
def fun1():
    print("in fun1")
    for i in range(8):
        time.sleep(0.3)

# return None 和没有return语句是一致的效果
print(fun0(9))
print(fun1())  # inner()

3. 装饰器工厂函数:

接受更多的参数,传递给内部函数使用,从而控制装饰器函数的行为。
它可以提供一个装饰器函数,即在装饰器外面再定义一个函数,让该函数来调用装饰器
装饰器工厂函数,带有(),及表示被调用执行,带参数给里面的装饰器,来操作该装饰器

import time

def get_run_time(flag):
    def get_time(func):
        def inner(*args, **kwargs):
            # args接收所有的位置参数 放到args中
            # kwargs接收所有的命名参数 放到kwargs
            begin = time.time()
            # 执行被装饰的函数的 功能  *args **kwargs代表解包、拆包
            ret = func(*args, **kwargs)
            end = time.time()
            if flag == 'int':
                print("花费了%ds" % int(end - begin))
            else:
                print("花费了%fs" % (end-begin))
            return ret
        return inner
    # 装饰器工厂函数的特点 内部函数是一个装饰器函数
    return get_time


# get_run_time('int') func0 = get_time(fun0)
@get_run_time('int')
def fun0(number):
    print("in fun0 %d" % number)
    return 100


@get_run_time('float')
def fun1(number):
    print("in fun1 %d" % number)
    return 100

fun0(1)
fun1(99)

4. 类装饰器

装饰器函数其实是这样一个接口约束,它必须接受一个callable对象作为参数,然后返回一个callable对象。在Python中一般callable对象都是函数,但也有例外。只要某个对象重写了 __call__() 方法,那么这个对象就是callable的。

class Test():
    def __call__(self):
        print('call me!')

t = Test()
t()  # call me

类装饰器demo

class Test(object):
    def __init__(self, func):
        print("---初始化---")
        print("func name is %s"%func.__name__)
        self.__func = func
    def __call__(self):
        print("---装饰器中的功能---")
        self.__func()
#说明:
#1. 当用Test来装作装饰器对test函数进行装饰的时候,首先会创建Test的实例对象
#    并且会把test这个函数名当做参数传递到__init__方法中
#    即在__init__方法中的func变量指向了test函数体
#
#2. test函数相当于指向了用Test创建出来的实例对象
#
#3. 当在使用test()进行调用时,就相当于让这个对象(),因此会调用这个对象的__call__方法
#
#4. 为了能够在__call__方法中调用原来test指向的函数体,所以在__init__方法中就需要一个实例属性来保存这个函数体的引用
#    所以才有了self.__func = func这句代码,从而在调用__call__方法中能够调用到test之前的函数体
@Test
def test():
    print("----test---")
test()
showpy()#如果把这句话注释,重新运行程序,依然会看到"--初始化--"

运行结果如下:

---初始化---
func name is test
---装饰器中的功能---
----test---
  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值