资讯专栏INFORMATION COLUMN

Python装饰器以及高级用法

impig33 / 1859人阅读

摘要:装饰器的高级用法介绍下面这些旨在介绍装饰器的一些更有趣的用法。装饰器在定义时向函数和方法添加功能,它们不用于在运行时添加功能。接受参数的装饰器有时,除了装饰的函数之外,装饰器还可以使用参数。

介绍

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

用例:计时函数执行

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

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

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

def func_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都要写四行代码而不是一行,这是非常好的

好的,但是只需要计算一个函数的时间是不现实的。如果你需要对一件事进行计时,你很有可能需要至少对两件事进行计时。所以我们会选择三个。

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

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

def func_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_afunc_bfunc_c中的方法,这种方法不会让我们到处复制粘贴代码。

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

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

def get_function():
    print ("inside get_function")                 
    def returned_function():                    
        print("inside returned_function")        
        return 1
    print("outside returned_function")
    return returned_function

returned_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_function
outside returned_function

此时Python不执行returned_function的任何内容。

C

这一行输出:

也就是说,get_function()返回的值x本身就是一个函数。

尝试再次运行B和C行。请注意,每次重复此过程时,返回的returned_function地址都是不同。每次调用get_function都会生成新的returned function

d

因为x是函数,所以就可以调用它。调用x就是调用returned_function的一个实例。这里输出的是:

inside returned_function
1

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

回到时间问题

你现在仍然在看么?如此我们有了新的知识,那么我们如何解决我们的老问题?我建议我们创建一个函数,让我们调用它并称为time_this,它将接收另一个函数作为参数,并将参数函数封装在某些计时代码中。有点像:

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

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

1这只是time_this的原型。time_this是一个函数就像任何其他函数一样,并且只有一个参数。 2我们在内部定义一个函数time_this。每当time_this执行时它都会创建一个新函数。 3计时代码,就像之前一样。 4我们调用原始函数并保留结果以供日后使用。 5,6剩余的计时代码。 7new_function必须像原始函数一样运行,因此返回存储的结果。 8返回在time_this中创建的函数。

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

def func_a(stuff):
    do_important_things_1()
    do_important_things_2()
    do_important_things_3()
func_a = time_this(func_a)        # <---------

def func_b(stuff):
    do_important_things_4()
    do_important_things_5()
    do_important_things_6()
func_b = time_this(func_b)        # <---------

def func_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_this
def func_a(stuff):
    do_important_things_1()
    do_important_things_2()
    do_important_things_3()

完全等同于:

def func_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的,因此这是一件非常简单的事情。

一个基本的装饰

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

def time_this(original_function):      
    def new_function(*args,**kwargs):
        import datetime                 
        before = datetime.datetime.now()                     
        x = original_function(*args,**kwargs)                
        after = datetime.datetime.now()                      
        print ("Elapsed Time = {0}".format(after-before))    
        return x                                             
    return new_function                                   

@time_this
def func_a(stuff):
    import time
    time.sleep(3)

func_a(1)
接受参数的装饰器

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

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

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

#这些功能是存在的
def current_user_id():
    """
    此函数返回当前登录的用户ID,如果没有经过身份验证,则返回None 
    """

def get_permissions(iUserId):
    """
    返回给定用户的权限字符串列表,例如 ["logged_in","administrator","premium_member"]
    """

#我们需要对这些函数进行权限检查

def delete_user(iUserId):
   """
   删除具有给定ID的用户,只有管理员权限才能访问此函数
   """

def new_game():
    """
    任何已登录的用户都可以启动一个新游戏
    """

def premium_checkpoint():
   """
   保存游戏进程,只允许高级成员访问
   """

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

def requires_admin(fn):
    def ret_fn(*args,**kwargs):
        lPermissions = get_permissions(current_user_id())
        if "administrator" in lPermissions:
            return fn(*args,**kwargs)
        else:
            raise Exception("Not allowed")
    return ret_fn

def requires_logged_in(fn):
    def ret_fn(*args,**kwargs):
        lPermissions = get_permissions(current_user_id())
        if "logged_in" in lPermissions:
            return fn(*args,**kwargs)
        else:
            raise Exception("Not allowed")
    return ret_fn

def requires_premium_member(fn):
    def ret_fn(*args,**kwargs):
        lPermissions = get_permissions(current_user_id())
        if "premium_member" in lPermissions:
            return fn(*args,**kwargs)
        else:
            raise Exception("Not allowed")
    return ret_fn

@requires_admin
def delete_user(iUserId):
   """
   删除具有给定Id的用户,只有具有管理员权限的用户才能访问此函数
   """

@requires_logged_in 
def new_game():
    """
    任何已登录的用户都可以启动一个新游戏
    """

@requires_premium_member
def premium_checkpoint():
   """
   保存游戏进程,只允许高级成员访问
   """

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

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

def requires_permission(sPermission):                            
    def decorator(fn):                                            
        def decorated(*args,**kwargs):                            
            lPermissions = get_permissions(current_user_id())     
            if sPermission in lPermissions:                       
                return fn(*args,**kwargs)                         
            raise Exception("permission denied")                  
        return decorated                                          
    return decorator       

def get_permissions(iUserId): #这样装饰器就不会抛出NameError
    return ["logged_in",]

def current_user_id():        #名称错误也是如此
    return 1

#现在我们可以进行装饰了                                 

@requires_permission("administrator")
def delete_user(iUserId):
   """
   删除具有给定Id的用户,只有具有管理员权限的用户才能访问此函数
   """

@requires_permission("logged_in")
def new_game():
    """
    任何已登录的用户都可以启动一个新游戏
    """

@requires_permission("premium_member")
def premium_checkpoint():
   """
   保存游戏进程,只允许高级成员访问
   """

尝试调用delete_usernew_gamepremium_checkpoint看看会发生什么。

premium_checkpointdelete_user都在消息“权限被拒绝”的情况下引发异常,new_game执行得很好(但没有太多的作用)。

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

def outer_decorator(*outer_args,**outer_kwargs):                            
    def decorator(fn):                                            
        def decorated(*args,**kwargs):                            
            do_something(*outer_args,**outer_kwargs)                      
            return fn(*args,**kwargs)                         
        return decorated                                          
    return decorator       

@outer_decorator(1,2,3)
def foo(a,b,c):
    print (a)
    print (b)
    print (c)

foo()

这相当于:

def decorator(fn):                                            
    def decorated(*args,**kwargs):                            
        do_something(1,2,3)                      
        return fn(*args,**kwargs)                         
    return decorated                                          
return decorator       

@decorator
def foo(a,b,c):
    print (a)
    print (b)
    print (c)

foo()
装饰课程

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

class ImportantStuff(object):
    @time_this
    def do_stuff_1(self):
        ...
    @time_this
    def do_stuff_2(self):
        ...
    @time_this
    def do_stuff_3(self):
        ...

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

@time_all_class_methods
class ImportantStuff:
    def do_stuff_1(self):
        ...
    def do_stuff_2(self):
        ...
    def do_stuff_3(self):
        ...

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

class ImportantStuff:
    def do_stuff_1(self):
        ...
    def do_stuff_2(self):
        ...
    def do_stuff_3(self):
        ...

ImportantStuff = time_all_class_methods(ImportantStuff)

那么time_all_class_methods是如何工作的? 首先,我们知道它需要将一个类作为参数,并返回一个类。我们也知道返回类的函数应该与原始ImportantStuff类的函数相同。也就是说,我们仍然希望想要完成重要的事情,我们需要进行计时。以下是我们将如何做到这一点:

def time_this(original_function):      
    print ("decorating")                      
    def new_function(*args,**kwargs):
        print ("starting timer")      
        import datetime                 
        before = datetime.datetime.now()                     
        x = original_function(*args,**kwargs)                
        after = datetime.datetime.now()                      
        print ("Elapsed Time = {0}".format(after-before))      
        return x                                             
    return new_function  

def time_all_class_methods(Cls):
    class NewCls(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)
            except AttributeError:      
                pass
            else:
                return x
            x = self.oInstance.__getattribute__(s)
            if type(x) == type(self.__init__): # 这是一个实例方法
                return time_this(x)                 # 这等价于用time_this修饰方法
            else:
                return x
    return NewCls

#现在让我们做一个虚拟类来测试它:

@time_all_class_methods
class Foo(object):
    def a(self):
        print ("entering a")
        import time
        time.sleep(3)
        print ("exiting a")

oF = Foo()
oF.a()
结论

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

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

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/45051.html

相关文章

  • python高级特性

    摘要:常规的使用来统计一段代码运行时间的例子输出结果总结其实是一门特别人性化的语言,但凡在工程中经常遇到的问题,处理起来比较棘手的模式基本都有对应的比较优雅的解决方案。 python的高级特性 名词与翻译对照表 generator 生成器 iterator 迭代器 collection 集合 pack/unpack 打包/解包 decorator 装饰器 context manager ...

    yexiaobai 评论0 收藏0
  • Python装饰高级用法

    摘要:在中,装饰器一般用来修饰函数,实现公共功能,达到代码复用的目的。智能装饰器上节介绍的写法,嵌套层次较多,如果每个类似的装饰器都用这种方法实现,还是比较费劲的脑子不够用,也比较容易出错。假设有一个智能装饰器,修饰装饰器,便可获得同样的能力。 在Python中,装饰器一般用来修饰函数,实现公共功能,达到代码复用的目的。在函数定义前加上@xxxx,然后函数就注入了某些行为,很神奇!然而,这只...

    AlphaWallet 评论0 收藏0
  • python装饰详解

    摘要:为了避免重复调用,可以适当地做缓存,的装饰器可以完美的完成这一任务。这意味着我们可以为方法创建装饰器,只是要记得考虑。装饰器封装了函数,这使得调试函数变得困难。另外,使用装饰器去管理缓存和权限。 原文地址 之前用python简单写了一下斐波那契数列的递归实现(如下),发现运行速度很慢。 def fib_direct(n): assert n > 0, invalid n ...

    maybe_009 评论0 收藏0
  • Python装饰的另类用法

    摘要:今天我们一起探讨一下装饰器的另类用法。语法回顾开始之前我们再将装饰器的语法回顾一下。例子本身只是演示了装饰器的一种用法,但不是推荐你就这样使用装饰器。类装饰器在以前,还不支持类装饰器。 之前有比较系统介绍过Python的装饰器(请查阅《详解Python装饰器》),本文算是一个补充。今天我们一起探讨一下装饰器的另类用法。 语法回顾 开始之前我们再将Python装饰器的语法回顾一下。 @d...

    hqman 评论0 收藏0
  • 为什么离不开 Stackoverflow

    摘要:接下来手工实现了一个简单的装饰器原型,紧接着引入中的装饰器语法。最后还列出了一些装饰器的高级用法,包括给装饰器传递参数等。读完整个答案,一定能对装饰器有较深的理解,并且知道理解装饰器的思考过程。 作为一名程序员,如果没有听过 Stackoverflow,那么你最好去面壁思过一下。程序员最需要阅读的一本编程书籍(其实编程书留下这本就够了!): showImg(https://segmen...

    peixn 评论0 收藏0

发表评论

0条评论

最新活动
阅读需要支付1元查看
<