[关闭]
@Scrazy 2017-04-16T02:05:34.000000Z 字数 2880 阅读 1013

装饰器小酌

python

Python中的装饰器真是个好东西。熟悉我的人都知道,挺能喷的一个人。但是,技术不是喷出来的,前言写的再花,也不会让我的装饰器比别人的更有威力。哈哈哈

开撸

装饰器是 Python 提供的一个语法糖,即:一个函数接受另一个函数作为参数并返回一个新的函数

func = decorate(func)

不带参数的

  1. In [17]: from functools import wraps
  2. ...:
  3. ...: def log(func):
  4. ...: @wraps(func)
  5. ...: def wrapper(*args, **kw):
  6. ...: print("I'm a log ^*^")
  7. ...: result = func(*args, **kw)
  8. ...: return result
  9. ...: return wrapper
  10. ...:
  11. In [19]: @log
  12. ...: def hello():
  13. ...: print('Hello World')
  14. ...:
  15. In [20]: hello()
  16. I'm a log ^*^
  17. Hello World

带参数的怎么写呢???

请看

  1. In [5]: from functools import wraps
  2. In [8]: def logs(file="info.log"):
  3. ...: def decorate(func):
  4. ...: @wraps(func)
  5. ...: def wrapper(*args, **kw):
  6. ...: log = func.__name__ + " was called"
  7. ...: print(log)
  8. ...: with open(file, 'a') as f:
  9. ...: f.write(log + '\n')
  10. ...: return wrapper
  11. ...: return decorate
  12. ...:
  13. In [11]: @logs()
  14. ...: def hello():
  15. ...: print('Hello World!')
  16. ...:
  17. In [12]: hello()
  18. hello was called
  19. In [13]: @logs(file='info2.log')
  20. ...: def hello2():
  21. ...: print('Hello World!')
  22. ...:
  23. In [14]: hello2()
  24. hello2 was called

带参数和不带参数

  1. from functools import wraps, partial
  2. import logging
  3. def logged(func=None, *, level=logging.DEBUG, name=None, message=None):
  4. if func is None:
  5. return partial(logged, level=level, name=name, message=message)
  6. logname = name if name else func.__module__
  7. log = logging.getLogger(logname)
  8. logmsg = message if message else func.__name__
  9. @wraps(func)
  10. def wrapper(*args, **kwargs):
  11. log.log(level, logmsg)
  12. return func(*args, **kwargs)
  13. return wrapper
  14. # Example use
  15. @logged
  16. def add(x, y):
  17. return x + y
  18. @logged(level=logging.CRITICAL, name='example') # 高于 warning 级别的才会输出到屏幕
  19. def spam():
  20. print('Spam!')

就到这里吧,反正很少有人会来看。就一个人浪吧。


更新


类装饰器

  1. In [1]: class Log():
  2. ...: def __init__(self, file="info.log"):
  3. ...: self.file = file
  4. ...: def __call__(self, func):
  5. ...: log = func.__name__ + " was called"
  6. ...: print(log)
  7. ...: with open(self.file, 'a') as f:
  8. ...: f.write(log+'\n')
  9. ...:
  10. In [2]: @Log()
  11. ...: def hello():
  12. ...: print('Hello World!')
  13. ...:
  14. hello was called

可复制粘贴代码在这里

缓存

  1. # coding: utf-8
  2. import functools
  3. def cache(func):
  4. _cache = {}
  5. functools.wraps(func)
  6. def wrapper(*args):
  7. result = _cache.get(args)
  8. if not result:
  9. result = func(*args)
  10. _cache[args] = result
  11. return result
  12. return wrapper
  13. # test
  14. In [5]: @cache
  15. ...: def fib(n):
  16. ...: if n < 2:
  17. ...: return n
  18. ...: return fib(n-1) + fib(n-2)
  19. ...:
  20. In [6]: %time fib(200)
  21. CPU times: user 0 ns, sys: 0 ns, total: 0 ns
  22. Wall time: 723 µs
  23. Out[6]: 280571172992510140037611932413038677189525
  24. In [7]: %time fib(200)
  25. CPU times: user 0 ns, sys: 0 ns, total: 0 ns
  26. Wall time: 11.2 µs # 时间大幅减少
  27. Out[7]: 280571172992510140037611932413038677189525
  28. In [8]: %time fib(200)
  29. CPU times: user 0 ns, sys: 0 ns, total: 0 ns
  30. Wall time: 13.4 µs
  31. Out[8]: 280571172992510140037611932413038677189525

flask_login.login_required

flask_login.login_required 是也是一个装饰器,login_required 装饰器的主要作用就是让只有已登陆和认证过的用户才能继续调用被其装饰的视图(view)函数。下面是截取的 片段

  1. def login_required(func):
  2. @wraps(func)
  3. def decorated_view(*args, **kwargs):
  4. if request.method in EXEMPT_METHODS:
  5. return func(*args, **kwargs)
  6. elif current_app.login_manager._login_disabled:
  7. return func(*args, **kwargs)
  8. elif not current_user.is_authenticated:
  9. return current_app.login_manager.unauthorized()
  10. return func(*args, **kwargs)
  11. return decorated_view

参考

  1. 哦,对了,附上一个牛逼的链接,墙裂推荐不熟悉装饰器的小伙伴们看看。

  2. Python tips Decorates

  3. Python CookBook
添加新批注
在作者公开此批注前,只有你和作者可见。
回复批注