中国领先的IT技术网站
|
|
创建专栏

装饰器为什么难以理解?

无论项目中还是面试都离不开装饰器话题,装饰器的强大在于它能够在不修改原有业务逻辑的情况下对代码进行扩展,但为什么初学者对装饰器的理解如此困难,我认为本质上是对Python函数理解不到位,因为装饰器本质上还是函数。

作者:刘志军|2017-07-07 17:01

开发者大赛路演 | 12月16日,技术创新,北京不见不散


装饰器为什么难以理解?

无论项目中还是面试都离不开装饰器话题,装饰器的强大在于它能够在不修改原有业务逻辑的情况下对代码进行扩展,权限校验、用户认证、日志记录、性能测试、事务处理、缓存等都是装饰器的绝佳应用场景,它能够最大程度地对代码进行复用。

但为什么初学者对装饰器的理解如此困难,我认为本质上是对Python函数理解不到位,因为装饰器本质上还是函数。

函数的定义

理解装饰器前,需要明白函数的工作原理,我们先从一个最简单函数定义开始:

  1. def foo(num): 
  2.     return num + 1 

上面定义了一个函数,名字叫foo,也可以把 foo 可理解为变量名,该变量指向一个函数对象

调用函数只需要给函数名加上括号并传递必要的参数(如果函数定义的时候有参数的话)

  1. value = foo(3) 
  2. print(value) # 4 

变量名 foo 现在指向 函数对象,但它也可以指向另外一个函数。

  1. def bar(): 
  2.     print("bar") 
  3. foo = bar 
  4. foo() # bar 

函数作为返回值

在Python中,一切皆为对象,函数也不例外,它可以像整数一样作为其它函数的返回值,例如:

  1. def foo(): 
  2.     return 1 
  3.  
  4. def bar(): 
  5.     return foo 
  6.  
  7. print(bar()) # <function foo at 0x10a2f4140> 
  8.  
  9. print(bar()()) # 1  
  10. # 等价于 
  11. print(foo()) # 1 

bar() 的返回值是一个函数对象,所以我们可以继续对返回值进行调用,调用bar()()等价于调用 foo(),因为 变量 foo 指向的对象与 bar() 的返回值是同一个对象。

函数作为参数

函数还可以像整数一样作为函数的参数,例如:

  1. def foo(num): 
  2.     return num + 1 
  3.  
  4. def bar(fun): 
  5.     return fun(3) 
  6.  
  7. value = bar(foo) 
  8. print(value)  # 4 

函数 bar 接收一个参数,这个参数是一个可被调用的函数对象,把函数 foo 传递到 bar 中去时,foo 和 fun 两个变量名指向的都是同一个函数对象 。所以调用 fun(3) 相当于调用 foo(3)。

函数嵌套

函数不仅可以作为参数和返回值,函数还可以定义在另一个函数中,作为嵌套函数存在,例如:

  1. def outer(): 
  2.     x = 1 
  3.     def inner(): 
  4.         print(x) 
  5.     inner() 
  6.  
  7. outer() # 1 

inner做为嵌套函数,它可以访问外部函数的变量,调用 outer 函数时,发生了3件事:

  • 给 变量 x 赋值为1
  • 定义嵌套函数 inner,此时并不会执行 inner 中的代码,因为该函数还没被调用,直到第3步
  • 调用 inner 函数,执行 inner 中的代码逻辑。

闭包

再来看一个例子:

  1. def outer(x): 
  2.     def inner(): 
  3.         print(x) 
  4.  
  5.     return inner 
  6. closure = outer(1) 
  7. closure() # 1 

同样是嵌套函数,只是稍改动一下,把局部变量 x 作为参数了传递进来,嵌套函数不再直接在函数里被调用,而是作为返回值返回,这里的 closure就是一个闭包,本质上它还是函数,闭包是引用了自由变量(x)的函数(inner)。

装饰器

继续往下看:

  1. def foo(): 
  2.     print("foo") 

上面这个函数这可能是史上最简单的业务代码了,虽然没什么用,但是能说明问题就行。现在,有一个新的需求,需要在执行该函数时加上日志:

  1. def foo(): 
  2.     print("记录日志开始") 
  3.     print("foo") 
  4.     print("记录日志结束") 

功能实现,唯一的问题就是它需要侵入到原来的代码里面,把日志逻辑加上去,如果还有好几十个这样的函数要加日志,也必须这样做,显然,这样的代码一点都不Pythonic。那么有没有可能在不修改业务代码的提前下,实现日志功能呢?答案就是装饰器。

  1. def outer(func): 
  2.     def inner(): 
  3.         print("记录日志开始") 
  4.         func() # 业务函数 
  5.         print("记录日志结束") 
  6.     return inner 
  7.  
  8. def foo(): 
  9.     print("foo") 
  10.  
  11. foo = outer(foo)  
  12. foo() 

我没有修改 foo 函数里面的任何逻辑,只是给 foo 变量重新赋值了,指向了一个新的函数对象。最后调用 foo(),不仅能打印日志,业务逻辑也执行完了。现在来分析一下它的执行流程。

这里的 outer 函数其实就是一个装饰器,装饰器是一个带有函数作为参数并返回一个新函数的闭包,本质上装饰器也是函数。outer 函数的返回值是 inner 函数,在 inner 函数中,除了执行日志操作,还有业务代码,该函数重新赋值给 foo 变量后,调用 foo() 就相当于调用 inner()

foo 重新赋值前:

重新赋值后:

另外,Python为装饰器提供了语法糖 @,它用在函数的定义处:

  1. @outer 
  2. def foo(): 
  3.     print("foo") 
  4.  
  5. foo() 

这样就省去了手动给foo重新赋值的步骤。

到这里不知你对装饰器理解了没有?当然,装饰器还可以更加复杂,比如可以接受参数的装饰器,基于类的装饰器等等。

【本文是51CTO专栏作者“刘志军”的原创文章,作者微信公众号:Python之禅(VTtalk)】

戳这里,看该作者更多好文

【编辑推荐】

  1. 外媒速递:Python编程学习当中必须回避的三种常见错误
  2. 人物|曲毅:曾经,我的幸福是坐在西湖边上敲代码
  3. R vs Python:R是现在最好的数据科学语言吗?
  4. 外媒速递:九大最强代码编辑器及其具体优势综述
  5. 代码这样写不止于优雅(Python版)
【责任编辑:IT疯 TEL:(010)68476606】

点赞 0
分享:
大家都在看
猜你喜欢

热门职位+更多