Python装饰器——初识

上次讲了Python的闭包,今天来讲一下闭包的应用——装饰器

1. 装饰器是什么

什么叫装饰器?顾名思义,它是一个用来装饰的东西。用来装饰谁呢?用来装饰函数。由于装饰器器自身也是一个函数,所以,一句话来说,装饰器是一个用来装饰函数的函数。感觉有点拗口,那么,在我的理解,装饰器只是个辅助函数,有没有它并不影响被装饰函数的运行。下面听我娓娓道来。

2.函数的表达

在讲之前,我们先来看一下函数,有这样一个函数:

def func():
    print "running func"

我们都知道,运行func()将会输出:

>> running func

然而,运行func,则输出:

>> <function func at 0x7f70236215f0>

所以我们看到在函数名之后加了括号会进入函数内部运行,而不加括号只是代表了一个函数对象(Python中一切皆对象,函数也是),记住这点可以帮我们更好的理解装饰器。

3.场景

现在我有一堆函数例如:

def func1():
    print "running func1"

def func2():
    print "running func2"

def func3():
    print "running func3"

不过现在,为了证明这些函数是我写的,都要打印一句话。那么,我可能会这样写:

def func1():
    print "running func1"
    print "强哥好帅!"

其他函数也是如此。然而,到以后函数多了的时候,你会感觉这样写很是麻烦,而且还更改了原函数的业务逻辑,非常不好。这时候,便可以用闭包来实现。

4.用闭包简化

改成闭包的形式如下:

#把此函数写在最上面,因为`Python`解释器自上而下解释
def shoe_me(func):
    def wrapper():
        print "强哥好帅!"
        func()
    return wrapper

#原函数
def func1():
    print "running func1"

func1 = show_me(func1)

if __name__ == '__main__':
    func1()

运行后的结果如图:

闭包实现

注意:当程序运行到func1 = show_me(func1)时并没有立即执行函数,只是定义了一个新函数。show_me(func1)其实等价于如下:

def wrapper():
    print "强哥好帅!"
    func1()

所以,只有当func1()时,才真正运行。这正好对应了第二部分中函数执行的原理。

5.用装饰器进一步简化

以上需求的实现还是有一点繁琐,Python大发给我们提供了@语法糖,来看一下怎么使用:

@shoe_me
def func2():
    print "running func2"

运行程序func2(),输出如下图:

装饰器实现

如程序所示,@shoe_me和原来闭包实现func2 = show_me(func2)是完全等价的,这样便能在不改变原函数的情况下添加装饰,增加功能。

显然,上面的例子除了装B,确实没有其他功能。那装饰器到底有什么实际功能呢?如函数日志分析(对上面的装饰器改进),登陆访问(Django中的@login_required),权限验证,缓存设置,记数器等,下一节将继续介绍装饰器的高级用法。

6.后记

略啰嗦,最后一部分才引入装饰器。。。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 我对Python装饰器的理解就是用一个函数去修饰另外一个函数。 装饰器 在运行原来功能的基础上,添加一些新的功能...
    敢梦敢当阅读 1,800评论 0 1
  • 呵呵!作为一名教python的老师,我发现学生们基本上一开始很难搞定python的装饰器,也许因为装饰器确实很难懂...
    TypingQuietly阅读 19,698评论 26 186
  • 本文为《爬着学Python》系列第四篇文章。从本篇开始,本专栏在顺序更新的基础上,会有不规则的更新。 在Pytho...
    SyPy阅读 7,194评论 4 11
  • 陈旧的木质桌子已掉了漆 发霉的食物沉默呻吟着 摆用多年的家电早就罢工不干 剩余的几块破布也成了珍宝 贫穷是何其卑劣...
    楚浅黛阅读 1,852评论 13 7
  • 学习API
    xunuo0x阅读 6,932评论 0 1