我记得5年前来搜狐面试时一个问题就是关于装饰器如何保持函数签名的问题。
由来
xadmin通过实现自己的BaseAdminView(继承自Django的View)来完成xadmin后台界面的处理。在解决一个csrf的问题时,翻了下xadmin BaseAdminView和Django的View部分的代码,关键点少了一条 update_wrapper
使用。导致我的小伙伴调试了半天。
还是看代码
可以对比看下:
## xadmin代码
@classonlymethod
def as_view(cls):
def view(request, *args, **kwargs):
self = cls(request, *args, **kwargs)
if hasattr(self, 'get') and not hasattr(self, 'head'):
self.head = self.get
if self.request_method in self.http_method_names:
handler = getattr(
self, self.request_method, self.http_method_not_allowed)
else:
handler = self.http_method_not_allowed
return handler(request, *args, **kwargs)
# take name and docstring from class
update_wrapper(view, cls, updated=())
view.need_site_permission = cls.need_site_permission
return view
### django/views/generic/base.py:class View中的代码
@classonlymethod
def as_view(cls, **initkwargs):
"""Main entry point for a request-response process."""
for key in initkwargs:
if key in cls.http_method_names:
raise TypeError("You tried to pass in the %s method name as a "
"keyword argument to %s(). Don't do that."
% (key, cls.__name__))
if not hasattr(cls, key):
raise TypeError("%s() received an invalid keyword %r. as_view "
"only accepts arguments that are already "
"attributes of the class." % (cls.__name__, key))
def view(request, *args, **kwargs):
self = cls(**initkwargs)
if hasattr(self, 'get') and not hasattr(self, 'head'):
self.head = self.get
self.request = request
self.args = args
self.kwargs = kwargs
return self.dispatch(request, *args, **kwargs)
view.view_class = cls
view.view_initkwargs = initkwargs
# take name and docstring from class
update_wrapper(view, cls, updated=())
# and possible attributes set by decorators
# like csrf_exempt from dispatch
update_wrapper(view, cls.dispatch, assigned=()) ## 差了这个啊!同学们
return view
xadmin是直接把dispatch的代码放到内部的view中了,这样看起来直观,但是缺少了对外可配置(通过重写增加装饰器)的dispatch函数。导致无法对view进行csrf_exempt装饰(其实是dispatch上装饰@csrf_exempt时,装饰器返回的inner上会设置csrf_exempt = True的属性)。
update_wrapper的用法
上述代码应该挺明显了,update_wrapper的作用就是把cls.dispatch
上的所有属性全部赋值到装饰函数上,也就是代码中的 view
。
在Python中有几个库是“居家旅行”必备的,functools就是之一,其中的partial也十分有用,用法参考这里python中functools宝库下的partial。
再举个例子吧
上面的代码,如果不熟悉对应的内容,可能不太懂。
关于保持函数签名,functools提供了两个api,一个是update_wrapper,一个是wrap装饰器函数,但是wrap装饰器函数也是调用了update_wrapper。所以就看update_wrapper就行。
有一个面试题是这样的,写一个函数装饰器,用来缓存函数的值。
函数是这样的:
def exec(sql):
""" 从执行数据库查询 """
return conn.execute(sql)
为了避免同样的语句执行多次数据库查询,我们需要做一层缓存,在不改变原函数的情况下。
于是有了这样的代码:
def cache(func):
cached_dict = {}
def inner(*args, **kwargs):
key = repr(args, kwargs)
try:
return cached_dict[key]
except KeyError:
cached_dict[key] = func(*args, **kwargs)
return cached_dict[key]
inner.csrf_exempt = True
return inner
@cache
def execute_query(sql):
""" 从执行数据库查询 """
print 'hit db' # 插播一条,刚才有人在群里问如何判断是否缓存了,看这个就行了
return 'result' # conn.execute(sql) # 假设拿到了结果
通过装饰器中的cached_dict来缓存同一个sql的结果。
我们print出来函数名称以及执行几个语句
print execute_query
# 输出 <function inner at 0x1025de6e0>
print execute_query('select * from test')
# 输出 hit db
# 输出 result
print execute_query('select * from test')
# 输出 result
第一个print出来的结果是存在问题的,我们调用的是execute_query,然而输出的函数名确实inner。所以这就需要update_wrapper或者wrap这样的函数来把被装饰的函数的属性(包括名称,doc等)放到装饰的函数上。也就是让inner伪装为execute_query。
要怎么做呢?一开始的那段代码就是例子了。两个方法一个是直接在inner上增加@functools.wrap(func)装饰器,另外一个方法是在return inner之前,增加一行: functools.update_wrapper(inner, func)
,然后重新执行上面的代码,结果是什么?
参考
- from the5fire.com微信公众号:Python程序员杂谈