- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我正在从关于 thecodeship 的精彩教程中学习一些关于装饰器的知识但发现自己对一个例子感到很困惑。
首先给出一个简单的例子,然后解释什么是装饰器。
def p_decorate(func):
def func_wrapper(name):
return "<p>{0}</p>".format(func(name))
return func_wrapper
def get_text(name):
return "lorem ipsum, {0} dolor sit amet".format(name)
my_get_text = p_decorate(get_text)
print my_get_text("John")
现在这对我来说很有意义。装饰器只是函数的包装器。在这个人的解释中,他说装饰器是一个函数,它将另一个函数作为参数,生成一个新函数,并返回生成的函数以在任何地方使用。
现在等价于上面的代码是:
def p_decorate(func):
def func_wrapper(name):
return "<p>{0}</p>".format(func(name))
return func_wrapper
@p_decorate
def get_text(name):
return "lorem ipsum, {0} dolor sit amet".format(name)
print get_text("John")
我相信我理解在没有参数的情况下初始化装饰器的方式。如果我错了,请纠正我。
get_text
,因为p_decorate
返回一个函数func_wrapper
,我们最终得到了正确的语句get_text = func_wrapper
。对我来说重要的是第一个等效代码块,因为我看到并理解装饰器的行为方式。
让我很困惑的是下面的代码:
def tags(tag_name):
def tags_decorator(func):
def func_wrapper(name):
return "<{0}>{1}</{0}>".format(tag_name, func(name))
return func_wrapper
return tags_decorator
@tags("p")
def get_text(name):
return "Hello "+name
print get_text("John")
再一次,如果我错了请纠正我,但这是我的理解。
tags_decorator
假设将传入的参数是函数正在装饰,get_text
。以“非装饰器”形式查看等效代码块可能对我有帮助,但我似乎无法理解它开始时的样子。我也不明白为什么 tags_decorator
和 func_wrapper
都被返回。如果装饰器只需要返回 1 个函数来包装 get_text
,那么返回两个不同的函数的目的是什么。
作为旁注,它实际上归结为以下内容。
最佳答案
在限制范围内,执行 @
之后的所有内容以生成 装饰器。在您的第一个示例中, @
之后的只是一个名称:
@p_decorate
因此 Python 查找 p_decorate
并使用装饰函数作为参数调用它:
get_text = p_decorate(get_text)
(有点过分简化了,get_text
最初并未绑定(bind)到原始函数,但您已经掌握了要点)。
在你的第二个例子中,装饰器表达式有点复杂,它包括一个调用:
@tags("p")
所以 Python 使用 tags("p")
来寻找装饰器。最后执行的是:
get_text = tags("p")(get_text)
tags("p")
的输出 就是这里的装饰器!我将 tags
函数本身称为装饰器 factory,它会在调用时生成一个装饰器。当您调用 tags()
时,它会返回 tags_decorator()
。这才是真正的 装饰器。
您可以改为删除装饰器函数并对 "p"
值进行硬编码并直接使用它:
def tags_decorator_p(func):
def func_wrapper(name):
return "<{0}>{1}</{0}>".format("p", func(name))
return func_wrapper
@tags_decorator_p
def get_text(name):
# ...
但是您必须为 tags()
的参数的每个可能值创建单独的装饰器。这就是装饰器工厂的值(value),您可以向装饰器添加参数并更改函数的装饰方式。
装饰器工厂 可以接受任意数量的参数;它只是您调用以生成装饰器的函数。装饰器自身只能接受一个参数,即要装饰的函数。
我在回答开始时说,在限制范围内是有原因的; @
后面的表达式的语法只允许一个带点的名称(foo.bar.baz
,so 属性访问)和一个调用((arg1,arg2,关键字=arg3)
)。查看reference documentation .原文PEP 318状态:
The decorator statement is limited in what it can accept -- arbitrary expressions will not work. Guido preferred this because of a gut feeling [17] .
关于Python装饰器示例,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/36427374/
我有这些 ViewModel:RecordViewModel、ComponentViewModel,其中 RecordViewModel 本质上是几个 ComponentViewModel 的容器。
我正在尝试为我的 Controller 设置一个装饰器。我的目的是在我的应用程序中的所有 Controller 中引入一些常见的行为。 我已将其配置为在 Angular 1.2.x 中工作,但从 1.
我想用角上的时间戳装饰我生成的所有 JFreeCharts。 JFreeChart 框架中是否有一种方法可以在生成图表后在图像上绘制? 编辑:请注意,这些图表是在后台线程中生成并通过 servlet
在grails应用程序中,我想用自定义数据装饰每个日志。当前的需要只是在日志消息前添加当前用户名 我对如何解决此问题有一些想法: -Adding custom field to log4j patte
我有一个form-el,它只是一个容器,必须将所有子元素包装在具有特定类的div 中。我希望允许 from-el 来包装它们,而不是在每个表单元素中重复这个 div 。我可以循环所有元素并将它们包装在
我遵循以下约定来装饰 Python 类中的某些方法。我想知道是否有一些更好的方法可以做到同样的事情。我的方法看起来当然不太好;对原始成员函数的调用看起来一点也不直观。 from threading i
我的 DTO 看起来像这样 public class SomeDTO { public string last_name{ get; set; } public string ac
这可能是一个远景,但 .NET 的灵 active 一直让我惊叹不已,所以开始吧。 我正在开发一个 MVC 应用程序,它需要在一组程序集中搜索派生自公共(public)基类的类类型。 即我有几个程序集
在我正在进行的项目中,我正在按照项目负责人的要求实现开发人员通知系统。它的工作方式是,如果发生前端错误,开发团队会收到一封错误电子邮件。 但是,在我当前的实现中,我似乎有以下循环依赖: $rootSc
我需要一些关于如何取消装饰和装饰 JInternalFrame 的帮助。我的框架类是这样的: package com; import java.awt.BorderLayout; import jav
假设我们有可能需要长时间运行的任务: public class LongRunningTask { public ReturnType doSomething() { ...
我正在尝试创建一个装饰器方法,它将一些默认的生命周期方法添加到 react 组件中。我的目标是向组件中添加一些默认功能,例如,所有组件都应该能够在 componentWillMount 上执行特定操作
我正在尝试将 DBUS 的异步方法调用与 Twisted 的 Deferred 相结合,但我在调整常用的 DBUS 服务方法装饰器来执行此操作时遇到了麻烦。 要使用 DBUS 异步回调方法,您需要:
是否可以设置表格中一行的背景颜色?当条件适用时,我需要突出显示一行。效果为 ...我可以在其中指定“字体”属性。 (我需要突出显示整行)。 最佳答案 您必须子类化 qooxdoo 默认行渲染器才能做到
我正在开发一个具有不同视角的基于 Java Swing 的应用程序。对于“主菜单”视角,我不希望装饰窗口 (JFrame),而在其他视角中,我确实希望装饰窗口。换句话说,我需要动态更 retrofit
我想做一些类似下面代码所示的事情: class foo { private: std::fstream* m_stream; public: foo(std::fstream* str
我的数据源提供了一个 ObservableList ,但是对于我的 ListView,我需要一个 ObservableList . A Warning基本上只是字符串的装饰器,添加一个 boolean
我一直在纠结于装饰+接口(interface)。假设我有以下“行为”接口(interface): interface IFlyable { void Fly();} interface ISwimma
有没有人为 iOS 6 UICollectionView 实现过装饰 View ?不可能 查找有关在网络上实现装饰 View 的任何教程。基本上在我的应用程序中,我有多个部分,我只想在每个部分后面显示
我有一个简单的 Controller ,例如: function MyController($scope, $http) { ... $http.post(url).success(f
我是一名优秀的程序员,十分优秀!