Python装饰器与闭包
闭包是Python装饰器的基础。要理解闭包,先要了解Python中的变量作用域规则。
变量作用域规则
首先,在函数中是能访问全局变量的:
>>> a = 'global var' >>> def foo():
print(a) >>> foo()
global var
然后,在一个嵌套函数中,内层函数能够访问在外层函数中定义的局部变量:
>>> def foo():
a = 'free var'
def bar():
print(a)
return bar >>> foo()()
free var
闭包
上面的嵌套函数就是闭包。闭包是指延伸了作用域的函数,在其中能够访问未在函数定义体中定义的非全局变量。未在函数定义体中定义的非全局变量一般都是在嵌套函数中出现的。
上述示例中的变量a就是一个并未在函数bar中定义的非全局变量。对于bar来说,它有个专业名字,叫做自由变量。
自由变量的名称可以在字节码对象中查看:
>>> bar = foo()
>>> bar.__code__.co_freevars
('a',)
自由变量的值绑定在函数的__closure__属性中:
>>> bar.__closure__
(<cell at 0x000001CB2912DF48: str object at 0x000001CB291D3D70>,)
其中保存了对应自由变量的cell对象的序列,cell对象的cell_contents属性保存了变量的值:
>>> bar.__closure__[0].cell_contents
'free var'
这与JavaScript中闭包的行为是类似的,JavaScript中嵌套函数会将外层函数的活动对象添加到它的作用域链中。但与JavaScript不同的是,当Python函数中的全局变量或者自由变量是不可变对象(数字、字符串、元组等)时,是只能读取,无法更新的:
>>> a = 1
>>> def foo():
print(a)
a += 1 >>> foo()
UnboundLocalError: local variable 'a' referenced before assignment >>> def foo():
a = 1
def bar():
print(a)
a += 1
return bar >>> foo()()
UnboundLocalError: local variable 'a' referenced before assignment
两种情况下,都会报错。这并不是缺陷,而是Python的设计选择。Python不要求声明变量,但是会假定在函数定义体中赋值的变量是局部变量,以避免在不知情的情况下修改全局变量。
a += 1与a = a + 1相同,编译函数的定义体时,会将a当做局部变量,不会当做自由变量保存。然后尝试获取a的值时,发现a并没有绑定值,于是报错。
解决这个问题的办法,一是将变量置于一些可变对象,如列表、字典中:
def foo():
ns = {}
ns['a'] = 1
def bar():
ns['a'] += 1
print (ns['a'])
return bar
另外的方法就是使用global或者nonlocal将变量声明为全局变量或者自由变量:
>>> def foo():
a = 1
def bar():
nonlocal a
a += 1
print(a)
return bar >>> foo()()
2
当自由变量本身是可变对象时,是可以直接进行操作的:
def make_avg():
ls = []
def avg(x):
ls.append(x)
print(sum(ls)/len(ls))
return avg
装饰器
装饰器是可调用对象,参数一般是另一个函数。装饰器可以以某种方式增强被装饰函数的行为,然后返回被装饰的函数或者将其替换成一个新的函数。
一个最简单的不做任何额外行为的装饰器:
def decorate(func):
return func
decorate函数就是一个最简单的装饰器,使用方法:
def target():
pass target = decorate(target)
Python为装饰器的使用提供了语法糖,可以简便的写为:
@decorate
def target():
pass
导入时运行
装饰器一个很重要的特性是它是导入时(加载模块时)运行的:
def decorate(func):
print('running decorator when import')
return func @decorate
def foo():
print('running foo')
pass if __name__ == '__main__':
print('start foo')
foo()
结果:
running decorator when import
start foo
running foo
可以看到,装饰器是导入时运行的,而被装饰的函数是明确调用时运行的。
装饰器可以返回被装饰的函数本身,和运行时导入的特性结合起来,可以实现简单的注册器功能:
view_registry = [] def register(func):
view_registry.append(func)
return func @register
def view1():
pass @register
def view2():
pass def main():
print(view_registry) if __name__ == '__main__':
main()
返回新函数
上述装饰器的例子都返回了被装饰的原函数,但装饰器的典型行为还是返回一个新函数:把被装饰的函数替换成新函数,新函数接受与原函数相同的参数,并且返回原函数本该返回的值。写法类似于:
def deco(func):
def new_func(*args, **kwargs):
return func(*args, **kwargs)
return new_func
这种情况下装饰器就使用到了闭包。JavaScript中的防抖与节流函数就是这种典型的装饰器行为。新函数一般会使用外部装饰器函数中的变量当做自由变量,对函数作出某种增强行为。
举个例子,我们知道,当Python函数的参数是个可变对象时,会产生意料之外的行为:
def foo(x, y=[]):
y.append(x)
print(y) foo(1)
foo(2)
foo(3)
输出:
[1]
[1, 2]
[1, 2, 3]
这是因为,函数的参数默认值保存在__defaults__属性中,指向了同一个列表:
>>> foo.__defaults__
([1, 2, 3],)
我们就可以用一个装饰器在函数执行前取出默认值做深复制,然后覆盖函数原先的参数默认值:
import copy def fresh_defaults(func):
defaults = func.__defaults__
def deco(*args, **kwargs):
func.__defaults__ = copy.deepcopy(defaults)
return func(*args, **kwargs)
return deco @fresh_defaults
def foo(x, y=[]):
y.append(x)
print(y) foo(1)
foo(2)
foo(3)
输出:
[1]
[2]
[3]
接收参数的装饰器
装饰器除了可以接受函数作为参数外,还可以接受其他参数。使用方法是:创建一个装饰器工厂,接受参数,返回一个装饰器,再把它应用到被装饰的函数上,语法如下:
def deco_factory(*args, **kwargs):
def deco(func):
print(args)
return func
return deco @deco_factory('factory')
def foo():
pass
在Web框架中,通常要将URL模式映射到生成响应的view函数,并将view函数注册到某些中央注册处。之前我们曾经实现过一个简单的注册装饰器,只是注册了view函数,却没有URL映射,是远远不够的。
在Flask中,注册view函数需要一个装饰器:
@app.route('/hello')
def hello():
return 'Hello, World'
原理就是使用了装饰器工厂,可以简单的模拟一下实现:
class App:
def __init__(self):
self.view_functions = {} def route(self, rule):
def deco(view_func):
self.view_functions[rule] = view_func
return view_func
return deco app = App() @app.route('/')
def index():
pass @app.route('/hello')
def hello():
pass for rule, view in app.view_functions.items():
print(rule, ':', view.__name__)
输出:
/ : index
/hello : hello
还可以使用装饰器工厂来确定view函数可以允许哪些HTTP请求方法:
def action(methods):
def deco(view):
view.allow_methods = [method.lower() for method in methods]
return view
return deco @action(['GET', 'POST'])
def view(request):
if request.method.lower() in view.allow_methods:
...
重叠的装饰器
装饰器也是可以重叠使用的:
@d1
@d2
def foo():
pass
等同于:
foo = d1(d2(foo))
类装饰器
装饰器的参数也可以是一个类,也就是说,装饰器可以装饰类:
import types def deco(cls):
for key, method in cls.__dict__.items():
if isinstance(method, types.FunctionType):
print(key, ':', method.__name__)
return cls @deco
class Test:
def __init__(self):
pass def foo(self):
pass
Python装饰器与闭包的更多相关文章
- python装饰器执行顺序
. python 装饰器 1) 2层装饰器 def decorator(func): # TODO def wrapper(*args, **kwargs): # TODO func(*args, * ...
- python 装饰器、内部函数、闭包简单理解
python内部函数.闭包共同之处在于都是以函数作为参数传递到函数,不同之处在于返回与调用有所区别. 1.python内部函数 python内部函数示例: def test(*args): def a ...
- 《流畅的Python》第三部分 把函数视作对象 【一等函数】【使用一等函数实现设计模式】【函数装饰器和闭包】
第三部分 第5章 一等函数 一等对象 在运行时创建 能赋值给变量或数据结构中的元素 能作为参数传递给函数 能作为函数的返回结果 在Python中,所有函数都是一等对象 函数是对象 函数本身是 func ...
- Python装饰器,Python闭包
可参考:https://www.cnblogs.com/lianyingteng/p/7743876.html suqare(5)等价于power(2)(5):cube(5)等价于power(3)(5 ...
- python函数下篇装饰器和闭包,外加作用域
装饰器和闭包的基础概念 装饰器是一种设计模式能实现代码重用,经常用于查日志,性能测试,事务处理等,抽离函数大量不必的功能. 装饰器:1.装饰器本身是一个函数,用于装饰其它函数:2.功能:增强被装饰函数 ...
- 关于python装饰器
关于python装饰器,不是系统的介绍,只是说一下某些问题 1 首先了解变量作用于非常重要 2 其次要了解闭包 def logger(func): def inner(*args, **kwargs) ...
- Python装饰器详解
python中的装饰器是一个用得非常多的东西,我们可以把一些特定的方法.通用的方法写成一个个装饰器,这就为调用这些方法提供一个非常大的便利,如此提高我们代码的可读性以及简洁性,以及可扩展性. 在学习p ...
- 关于python装饰器(Decorators)最底层理解的一句话
一个decorator只是一个带有一个函数作为参数并返回一个替换函数的闭包. http://www.xxx.com/html/2016/pythonhexinbiancheng_0718/1044.h ...
- python 装饰器 一篇就能讲清楚
装饰器一直是我们学习python难以理解并且纠结的问题,想要弄明白装饰器,必须理解一下函数式编程概念,并且对python中函数调用语法中的特性有所了解,使用装饰器非常简单,但是写装饰器却很复杂.为了讲 ...
随机推荐
- const 变量在多个文件共享,如何验证两种不同的方式下,编译器是否会在多个文件下建立多个副本
对于const变量多个文件共享,当我们不希望编译器为每个文件分别生成独立的变量,而是像非常量对象一个,一处定义,多处声明并使用. 解决办法是,对于const变量,不管是声明还是定义都添加extern关 ...
- 2018-2019-2 网络对抗技术 20165230 Exp9 :Web安全基础
目录 实验目的 实验内容 Webgoat前期准备 出现的问题 (一)SQL注入攻击 命令注入:Command Injection 数字型注入:Numeric SQL Injection 日志欺骗:Lo ...
- Linux配置DNS
vi /etc/resolv.conf, 后面加上nameserver 114.114.114.114
- shell (二) shell for循环
for循环 基本示例 for i in var1 var2 var2 do echo $i done for i in {1..100} do echo $i done 列表中复杂值,可以使用引号或者 ...
- Windows Server实例防火墙策略的配置方法
概述 本文介绍在Windows Server实例中,如何配置防火墙策略的方法. 详细描述 配置Windows Server版本的防火墙功能方法,参考如下步骤. 提示:此处以Windows Server ...
- apt-get命令使用
1.apt-get命令 apt-get命令是Debian Linux发行版中的APT软件包管理工具,所有基于Debian的发行都使用这个包管理系统. (1)命令语法 apt-get(选项)(参数) ( ...
- 【Netcore】使用 Magic生成器 ,零代码实现CRUD - HTTP REST 之接口
软件介绍: Magic是一个CRUD后端生成器,内置于ASP.NET内核中.它的目的是让你“神奇地”做一些无聊的事情,通过使用自动化技术,创建80%的CRUD端点,自动包装MySQL或MS SQL S ...
- Reliable Multicast Programming(PGM)协议
Reliable Multicast Programming (PGM)实际通用可靠多播协议,在某种程度上保证多播的可靠性.是IP上层协议,和TCP还有UDP同级,工作在传输层. 在组播传输视频项目中 ...
- C基本语法
分号 ; 在C程序中,分好是语句结束符,每个语句必须以分好结束,它表明一个逻辑实体的结束 例如: printf("Hello, World! \n"); ; 注释 // 单行注释 ...
- 整理:WPF中XmlDataProvider的用法总结
原文:整理:WPF中XmlDataProvider的用法总结 一.目的:了解XmlDataProvider中绑定数据的方法 二.绑定方式主要有三种: 1.Xaml资源中内置: <!--XPath ...