python的装饰器是静态的,也就是说你使用的装饰器一定是定义好的对象。在某些特殊的情况下,需要让动态使用装饰器。
警告
不要随便用这个办法,一点都不优雅,能够使用其他的方法规避动态调用装饰器就不要这么写。
需求
装饰器是类的一个属性,初始化时传入一个特定的对象。某个类方法需要被这个特定实例装饰才能生效。
我的问题情景
类需要使用一个Flask实例初始化,类定义了一些列路由。这些路由需要被绑定在Flask实例上。
一个方法是使用add_url_rule()
添加视图函数。但是这样做没添加一个视图函数就会多写一条add_url_rule()
,并不是很优雅。比较优雅的方式是写一个dict,里面定义路由和视图函数,然后遍历dict完成添加。
思路
动态装饰器的问题无非是装饰器不能使用self访问类,那用嵌套函数不就可以访问了吗,就是丑了一点而已
class decorator_test:
def __init__(self):
self.test()
self.wrapped_func()
def wrapper(self, func):
def wrapped(*args, **kwargs):
print('wrapped')
return func(*args, **kwargs)
return wrapped
def test(self):
@self.wrapper
def wrapped_func():
print('wrapped_func')
self.__setattr__('wrapped_func', wrapped_func)
看看就好