python装饰器作用和功能_Python装饰器以及高级用法

96dda144ad345982abb8a8c1c1e021a8cbef8448.jpeg?token=0a4d19816636577f6bba52e632f2f581&s=156DF104027683C41C6E85840300608B

介绍

首先我要承认,装饰器非常难!你在本教程中看到的一些代码将会有一些复杂。大多数人在学习Python时都跟装饰器做过斗争,所以如果这对你来说很奇怪,不要感到沮丧,因为同样的大多数人都可以克服这种苦难。在本教程中,我将逐步介绍了解装饰器的过程。首先我假设你已经可以编写基本函数和基本类。如果你不能做这些事,那么我建议你在回到这里之前先学习如何去做到编写基本函数和基本类(除非你迷路了,在这种情况下你可以原谅)。

用例:计时函数执行

假设我们正在执行一段代码,执行时间比我们想的还要长一些。这段代码由一堆函数调用组成,我们确信这些调用中至少有一个调用构成了我们代码中的瓶颈。我们如何找到瓶颈?现在有一个解决方案,就是我们现在要关注的解决方案,就是对函数执行进行计时。

让我们从一个简单的例子开始。我们只有一个函数需要计时,

func_a

deffunc_a(stuff):do_important_things_1()do_important_things_2()do_important_things_3()

一种方法是将时钟代码放在每个函数调用周围。所以就像这样:

func_a(current_stuff)

看起来会更像这样:

before=datetime.datetime.now()func_a(current_stuff)after=datetime.datetime.now()print("Elapsed Time = {0}".format(after-before))

这样就可以了。但是如果我们有多次调用

func_a

并且我们想要为所有这些计时会发生什么呢?我们可以用计时代码包围

func_a

的每个调用,但是这样做也有不好的效果。它只准备编写一次计时代码。因此,我们将其放在函数定义中,而不是将其放在函数之外。

deffunc_a(stuff):before=datetime.datetime.now()do_important_things_1()do_important_things_2()do_important_things_3()after=datetime.datetime.now()print("Elapsed Time = {0}".format(after-before))

这种方法的好处是:

我们将代码放在一个地方,所以如果我们想要更改它(例如,如果我们想将经过的时间存储在数据库或日志中)那么我们只需要在一个地方而不是每一个函数调用中更改它我们不需要记住每次调用func_a都要写四行代码而不是一行,这是非常好的好的,但是只需要计算一个函数的时间是不现实的。如果你需要对一件事进行计时,你很有可能需要至少对两件事进行计时。所以我们会选择三个。

deffunc_a(stuff):before=datetime.datetime.now()do_important_things_1()do_important_things_2()do_important_things_3()after=datetime.datetime.now()print("Elapsed Time = {0}".format(after-before))deffunc_b(stuff):before=datetime.datetime.now()do_important_things_4()do_important_things_5()do_important_things_6()after=datetime.datetime.now()print("Elapsed Time = {0}".format(after-before))deffunc_c(stuff):before=datetime.datetime.now()do_important_things_7()do_important_things_8()do_important_things_9()after=datetime.datetime.now()print("Elapsed Time = {0}".format(after-before))

这看起来很糟糕。如果我们想要对8个函数进行计时的时候怎么办?然后我们决定将计时的信息存储在日志文件中。然后我们决定建立一个更好的数据库。我们这里需要的是将一种相同的代码合并到

func_a

func_b

func_c

中的方法,这种方法不会让我们到处复制粘贴代码。

一个简单的绕道:返回函数的函数

Python是一种非常特殊的语言,因为函数是第一类对象。这意味着一旦函数在作用域中被定义,它就可以传递给函数,赋值给变量,甚至从函数返回。这个简单的事实是使python装饰器成为可能的原因。查看下面的代码,看看你是否可以猜出标记为A,B,C和D的行会发生什么。

defget_function():print("inside get_function")defreturned_function():print("inside returned_function")return1print("outside returned_function")returnreturned_functionreturned_function()# A x=get_function()# B x# C x()# D

A

这一行给出了一个

NameError

并声明

returned_function

不存在。但我们只是定义了它,对吧?你在这里需要知道的是,它是在get_function的范围内定义的。也就是说,在get_function里面定义了它。它不是在get_function之外。如果这让你感到困惑,那么你可以尝试使用该

locals()

函数,并阅读Python的范围。

B

这行代码打印出以下内容:

inside get_functionoutside returned_function

此时Python不执行

returned_function

的任何内容。

C

这一行输出:

也就是说,

get_function()

返回的值

x

本身就是一个函数。

尝试再次运行B和C行。请注意,每次重复此过程时,返回的

returned_function

地址都是不同。每次调用

get_function

都会生成新的

returned function

d

因为

x

是函数,所以就可以调用它。调用

x

就是调用

returned_function

的一个实例。这里输出的是

insidereturned_function1

也就是说,它打印字符串,并返回值

1

回到时间问题

你现在仍然在看么?如此我们有了新的知识,那么我们如何解决我们的老问题?我建议我们创建一个函数,让我们调用它并称为

time_this

,它将接收另一个函数作为参数,并将参数函数封装在某些计时代码中。有点像:

deftime_this(original_function):# 1defnew_function(*args,**kwargs):# 2before=datetime.datetime.now()# 3x=original_function(*args,**kwargs)# 4after=datetime.datetime.now()# 5print("Elapsed Time = {0}".format(after-before))# 6returnx# 7returnnew_function()# 8

我承认它有点疯狂,所以让我们一行一行的看下去:

1这只是

time_this

的原型。

time_this

是一个函数就像任何其他函数一样,并且只有一个参数。 2我们在内部定义一个函数

time_this

。每当

time_this

执行时它都会创建一个新函数。 3计时代码,就像之前一样。 4我们调用原始函数并保留结果以供日后使用。 5,6剩余的计时代码。 7new_function必须像原始函数一样运行,因此返回存储的结果。 8返回在time_this中创建的函数。

现在我们要确保我们的函数是计时的:

deffunc_a(stuff):do_important_things_1()do_important_things_2()do_important_things_3()func_a=time_this(func_a)# <---------deffunc_b(stuff):do_important_things_4()do_important_things_5()do_important_things_6()func_b=time_this(func_b)# <---------deffunc_c(stuff):do_important_things_7()do_important_things_8()do_important_things_9()func_c=time_this(func_c)# <---------

看看

func_a

,当我们执行时

func_a = time_this(func_a)

我们用

time_this

返回的函数替换

func_a

。所以我们用一个函数替换

func_A

该函数执行一些计时操作(上面的第3行),将

func a

的结果存储在一个名为

x

的变量中(第4行),执行更多的计时操作(第5行和第6行),然后返回

func_a

返回的内容。换句话说

func_a

,仍然以相同的方式调用并返回相同的东西,它也只是被计时了。是不是感觉很整洁?

介绍装饰器

我们所做的工作很好,而且非常棒,但是很难看,非常难读懂。所以Python可爱的作者给了我们一种不同的,更漂亮的写作方式:

@time_thisdeffunc_a(stuff):do_important_things_1()do_important_things_2()do_important_things_3()

完全等同于:

deffunc_a(stuff):do_important_things_1()do_important_things_2()do_important_things_3()func_a=time_this(func_a)

这通常被称为语法糖。

@

没有什么神奇的。这只是一个已达成一致的惯例。沿着这条路上的某个地方决定了。

总结

装饰器只是一个返回函数的函数。如果这些东西看起来非常的 - 那么请确保以下主题对你有意义然后再回到本教程:

Python函数范围Python作为第一类对象(甚至可以查找lambda函数,它可能使它更容易理解)。另一方面,如果你对更多的话题感兴趣的话,你可能会发现:

如装饰类:python @add_class_functionality class MyClass: ...

具有更多参数的装饰器, 例如:python @requires_permission(name="edit") def save_changes(stuff): ...

下面就是我要介绍的高级装饰器的主题。

装饰器的高级用法

介绍

下面这些旨在介绍装饰器的一些更有趣的用法。具体来说,如何在类上使用装饰器,以及如何将额外的参数传递给装饰器函数。

装饰者与装饰者模式

装饰器模式是一种面向对象的设计模式,其允许动态地将行为添加到现有的对象当中。当你装饰对象时,你将以独立于同类的其他实例方式扩展它的功能。

Python装饰器不是装饰器模式的实现。Python装饰器在定义时向函数和方法添加功能,它们不用于在运行时添加功能。装饰器模式本身可以在Python中实现,但由于Python是Duck-teped的,因此这是一件非常简单的事情。

一个基本的装饰

这是装饰器可以做的一个非常基本的例子。我只是把它作为一个参考点。在继续之前,请确保你完全理解这段代码。

deftime_this(original_function):defnew_function(*args,**kwargs):importdatetimebefore=datetime.datetime.now()x=original_function(*args,**kwargs)after=datetime.datetime.now()print("Elapsed Time = {0}".format(after-before))returnxreturnnew_function@time_thisdeffunc_a(stuff):importtimetime.sleep(3)func_a(1)

接受参数的装饰器

有时,除了装饰的函数之外,装饰器还可以使用参数。这种技术经常用于函数注册等事情。一个著名的例子是Pyramid Web应用程序框架中的视图配置。例如:

@view_config(route_name='home',renderer='templates/mytemplate.pt')defmy_view(request):return{'project':'hello decorators'}

假设我们有一个应用程序,用户可以登录并与一个漂亮的gui(图形用户界面)进行交互。用户与gui的交互触发事件,而这些事件导致Python函数被执行。让我们假设有很多用户使用这个应用程序,并且他们有许多不同的权限级别。执行不同的功能需要不同的权限类型。例如,考虑以下功能:

#这些功能是存在的defcurrent_user_id():""" 此函数返回当前登录的用户ID,如果没有经过身份验证,则返回None """defget_permissions(iUserId):""" 返回给定用户的权限字符串列表,例如 ['logged_in','administrator','premium_member'] """#我们需要对这些函数进行权限检查defdelete_user(iUserId):""" 删除具有给定ID的用户,只有管理员权限才能访问此函数 """defnew_game():""" 任何已登录的用户都可以启动一个新游戏 """defpremium_checkpoint():""" 保存游戏进程,只允许高级成员访问 """

实现这些权限的一种方法是创建多个装饰器,例如:

defrequires_admin(fn):defret_fn(*args,**kwargs):lPermissions=get_permissions(current_user_id())if'administrator'inlPermissions:returnfn(*args,**kwargs)else:raiseException("Not allowed")returnret_fndefrequires_logged_in(fn):defret_fn(*args,**kwargs):lPermissions=get_permissions(current_user_id())if'logged_in'inlPermissions:returnfn(*args,**kwargs)else:raiseException("Not allowed")returnret_fndefrequires_premium_member(fn):defret_fn(*args,**kwargs):lPermissions=get_permissions(current_user_id())if'premium_member'inlPermissions:returnfn(*args,**kwargs)else:raiseException("Not allowed")returnret_fn@requires_admindefdelete_user(iUserId):""" 删除具有给定Id的用户,只有具有管理员权限的用户才能访问此函数 """@requires_logged_indefnew_game():""" 任何已登录的用户都可以启动一个新游戏 """@requires_premium_memberdefpremium_checkpoint():""" 保存游戏进程,只允许高级成员访问 """

但这太可怕了。它需要大量的复制粘贴,并且每个装饰器需要不同的名称,如果对权限的检查方式进行了任何更改,则必须更新每个装饰器。有一个装饰器可以完成这三个工作不是很好吗?

为此,我们需要一个返回装饰器的函数:

defrequires_permission(sPermission):defdecorator(fn):defdecorated(*args,**kwargs):lPermissions=get_permissions(current_user_id())ifsPermissioninlPermissions:returnfn(*args,**kwargs)raiseException("permission denied")returndecoratedreturndecoratordefget_permissions(iUserId):#这样装饰器就不会抛出NameErrorreturn['logged_in',]defcurrent_user_id():#名称错误也是如此return1#现在我们可以进行装饰了 @requires_permission('administrator')defdelete_user(iUserId):""" 删除具有给定Id的用户,只有具有管理员权限的用户才能访问此函数 """@requires_permission('logged_in')defnew_game():""" 任何已登录的用户都可以启动一个新游戏 """@requires_permission('premium_member')defpremium_checkpoint():""" 保存游戏进程,只允许高级成员访问 """

尝试调用

delete_user

new_game

premium_checkpoint

看看会发生什么。

premium_checkpoint

delete_user

都在消息“权限被拒绝”的情况下引发异常,

new_game

执行得很好(但没有太多的作用)。

下面是装饰器的一般形式,带有参数和使用说明:

defouter_decorator(*outer_args,**outer_kwargs):defdecorator(fn):defdecorated(*args,**kwargs):do_something(*outer_args,**outer_kwargs)returnfn(*args,**kwargs)returndecoratedreturndecorator@outer_decorator(1,2,3)deffoo(a,b,c):print(a)print(b)print(c)foo()

这相当于:

defdecorator(fn):defdecorated(*args,**kwargs):do_something(1,2,3)returnfn(*args,**kwargs)returndecoratedreturndecorator@decoratordeffoo(a,b,c):print(a)print(b)print(c)foo()

装饰课程

装饰器不仅限于对函数进行操作,它们也可以对类进行操作。比方说,我们有一个类可以做很多非常重要的事情,我们想要把它所做的一切都进行计时。然后我们可以使用

time_this

像以前一样使用装饰器:

classImportantStuff(object):@time_thisdefdo_stuff_1(self):...@time_thisdefdo_stuff_2(self):...@time_thisdefdo_stuff_3(self):...

这样就可以了。但是这个类中还有一些额外的代码行。如果我们写一些更多的类方法并忘记装饰它们中的一个呢?如果我们决定不再为进行计时怎么办?这里肯定存在人为错误的空间。这样编写它会好得多:

@time_all_class_methodsclassImportantStuff:defdo_stuff_1(self):...defdo_stuff_2(self):...defdo_stuff_3(self):...

如你所知,该代码相当于:

classImportantStuff:defdo_stuff_1(self):...defdo_stuff_2(self):...defdo_stuff_3(self):...ImportantStuff=time_all_class_methods(ImportantStuff)

那么

time_all_class_methods

是如何工作的? 首先,我们知道它需要将一个类作为参数,并返回一个类。我们也知道返回类的函数应该与原始

ImportantStuff

类的函数相同。也就是说,我们仍然希望想要完成重要的事情,我们需要进行计时。以下是我们将如何做到这一点:

deftime_this(original_function):print("decorating")defnew_function(*args,**kwargs):print("starting timer")importdatetimebefore=datetime.datetime.now()x=original_function(*args,**kwargs)after=datetime.datetime.now()print("Elapsed Time = {0}".format(after-before))returnxreturnnew_functiondeftime_all_class_methods(Cls):classNewCls(object):def__init__(self,*args,**kwargs):self.oInstance=Cls(*args,**kwargs)def__getattribute__(self,s):""" 每当访问NewCls对象的任何属性时,都会调用这个函数。这个函数首先尝试 从NewCls获取属性。如果失败,则尝试从self获取属性。oInstance(一个 修饰类的实例)。如果它设法从self获取属性。oInstance, 属性是一个实例方法,然后应用' time_this '。 """try:x=super(NewCls,self).__getattribute__(s)exceptAttributeError:passelse:returnxx=self.oInstance.__getattribute__(s)iftype(x)==type(self.__init__):# 这是一个实例方法returntime_this(x)# 这等价于用time_this修饰方法else:returnxreturnNewCls#现在让我们做一个虚拟类来测试它:@time_all_class_methodsclassFoo(object):defa(self):print("entering a")importtimetime.sleep(3)print("exiting a")oF=Foo()oF.a()

结论

在装饰器的高级用法中,我向你展示了使用Python装饰器的一些技巧 - 我已经向你展示了如何将参数传递给装饰器,以及如何装饰类。但这仍然只是冰山的一角。在各种奇怪的情况下,有大量的方法用于装饰器。你甚至可以装饰你的装饰器(但如果你到达那一点,那么做一个全面的检查可能是个好主意)。Python同时内置了一些值得了解的装饰器,例如装饰器

staticmethod

classmethod

接下来要怎么做?除了我在这篇文章中向你展示的内容外,通常不需要对装饰器执行任何更复杂的操作。如果你对更改类功能的更多方法感兴趣,那么我建议阅读有关继承和一般OO设计原则的数据。或者,如果你真的想学会他们,那么请阅读元类(但同样,处理这些东西几乎不需要)。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值