类级别变量上的 Python 装饰器作为反射的元数据?

标签 python reflection decorator python-decorators

我有一个 python 类(实际上它是一个 Google App Engine 模型),如下所示:

class MyClass(ParentClass):
    variable_one = PropertyClass(...)
    variable_two = PropertyClass(...)

我打算构建一个系统,该系统可以描述此类的两个实例之间的差异,这两个实例表示实体在不同时间点的两个版本。最终它会产生一个像

这样的字符串
"Variable One changed from 'foo' to 'bar', Variable Two changed from empty to 'baz'"

因此,我需要在某处存储“variable_one”应显示为“Variable One”等事实。来自 C# 背景,我的直觉是使用属性,例如:

class MyClass : ParentClass
{
    [DisplayName("Variable One")]
    public PropertyClass variable_one;

    [DisplayName("Variable Two")]
    public PropertyClass variable_two;
}

起初,python 的等价物似乎是装饰器:

class MyClass(ParentClass):
    @display_name("Variable One")
    variable_one = PropertyClass(...)

    @display_name("Variable Two")
    variable_two = PropertyClass(...)

但是,装饰器实际上是在进行函数柯里化(Currying),并且由于类变量定义只是将 PropertyClass 的一个实例分配给 variable_one 和 variable_two,所以看起来这不是有效的 python。无论如何,我不确定我会怎么写。

一个选项似乎是扩展 PropertyClass...

class MyClass(ParentClass):
    variable_one = MyPropertyClass("Variable One", ...)
    variable_two = MyPropertyClass("Variable Two", ...)

但是有几个不同的属性类正在使用,我想为所有属性类添加一个共同的行为,所以这看起来太费力了。

我想我可以定义一个字典:

class MyClass(ParentClass):
    variable_one = PropertyClass(...)
    variable_two = PropertyClass(...)
    variable_display_names = {
        "variable_one": "Variable One",
        "variable_two": "Variable Two"
    }

但我希望听到一些更有经验的 python 爱好者关于实现这一目标的最佳 pythonic 方式。

最佳答案

数据存储属性类是描述符(如 python @property 装饰器),但作为不实现 __setattr__ 的自定义类,您可以在这些对象上设置任意属性:

 class MyClass(ParentClass):
     variable_one = PropertyClass(...)
     variable_one.display_name = "Variable One"

     variable_two = PropertyClass(...)
     variable_two.display_name = "Variable Two"

可以概括为装饰器:

def display_name(dispname):
    def decorate_with_display_name(prop):
        prop.display_name = dispname
        return prop
    return decorate_with_display_name

只要您通过类(而不是实例)访问属性,您就可以再次检索这些值;例如,给定一个实例,您可以执行 print type(instance).variable_two.display_name

关于类级别变量上的 Python 装饰器作为反射的元数据?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/14603481/

相关文章:

c# - cocoa 的属性网格

c++ - Qt4/PyQt4 - 无法为 QTextDocument 设置默认字体

Objective-C 类 -> 像 : [NSArray className] -> @"NSArray" 这样的字符串

python - 用python解析JSON : blank fields

c# - 如何创建返回 bool 值的 Lambda 表达式的 C# 反射

java - Java中的@override和Python中的@decorator的区别

python - Python 对象的唯一表示

c# - 当 bool 更改为 true 时,为什么我的自定义装饰器/装饰器不起作用?

python - 从 MySql 数据库中删除条目 Python Flask

python - 复选框形式的 Django 权限