通过 Python 装饰器实现DRY(不重复代码)原则
英文原文:DRY Principles through Python Decorators
Python装饰器是一个消除冗余的强大工具。随着将功能模块化为大小合适的方法,即使是最复杂的工作流,装饰器也能使它变成简洁的功能。 例如让我们看看Django web框架,该框架处理请求的方法接收一个方法对象,返回一个响应对象:
我最近遇到一个案例,需要编写几个满足下述条件的api方法:
做为一个注册api端点例子,我将会像这样编写:
| |
对于非post请求返回错误现在让我们在一些更有用的场景下应用装饰器。如果在django中接收的不是POST请求,我们用装饰器返回一个错误响应。
| |
发送json响应为了发送json响应(同时处理500状态码),我们可以新建另外一个装饰器:
|
BONUS: 参数化你的请求方法我曾经使用过Tubogears框架,其中请求参数直接解释转递给方法这一点我很喜欢。所以要怎样在Django中模仿这一特性呢?嗯,装饰器就是一种解决方案! 例如:
注意这是一个参数化装饰器的例子。在这个例子中,函数的结果是实际的装饰器。 现在我就可以用参数化装饰器编写方法了!我甚至可以选择是否允许GET和POST,或者仅仅一种请求参数类型。
|
BONUS #2: 使用functools.wraps保存docstrings和函数名很不幸,使用装饰器的一个副作用是没有保存方法名(__name__)和docstring(__doc__)值:
这将对使用反射的应用造成麻烦,比如Sphinx,一个 自动生成文档的应用。 为了解决这个问题,我们可以使用'wraps'装饰器附加上名字和docstring:
|
BONUS #3: 使用'decorator'装饰器如果仔细看看上述使用装饰器的方式,在包装器声明和返回的地方也有不少重复。 你可以安装python egg 'decorator',其中包含一个提供装饰器模板的'decorator'装饰器! 使用easy_install:
或者Pip:
然后你可以简单的编写:
|
本文中的所有译文仅用于学习和交流目的,转载请务必注明文章译者、出处、和本文链接
我们的翻译工作遵照 CC 协议,如果我们的工作有侵犯到您的权益,请及时联系我们
我们的翻译工作遵照 CC 协议,如果我们的工作有侵犯到您的权益,请及时联系我们