【问题标题】:Lazy class property decorator懒惰类属性装饰器
【发布时间】:2013-08-19 20:05:50
【问题描述】:

我有一个 django 模型,需要参考自定义用户模型进行一些处理。

由于类的加载顺序未知,我无法在类加载时使用此模型的类。

所以我需要在运行时添加一些类属性,目前我正在将它们添加到__init____new__ 中,例如:

def __new__(cls, *args, **kwargs):
    # hack to avoid INSTALLED_APPS initialization conflicts.
    # get_user_model() can't be called from this module at class loading time,
    # so some class attributes must be added later.
    # Metaclasses could me more appropiate but I don't want to override
    # dango's metaclasses.
    if not hasattr(cls, '_reverse_field_name_to_user'):
        cls._find_reverse_field_name_to_user()
    return Group.__new__(cls, *args, **kwargs)

它可以工作,但看起来很糟糕,所以我考虑过为这些属性使用类似 @lazyclassproperty 的东西。

我找到了几个 @classproperty@lazyproperty 装饰器,但没有一个适合两者,我不知道如何自己编写一个。

问题:我如何编写这样的装饰器?或者建议我目前愚蠢的实施的另一种更清洁的替代方案。

【问题讨论】:

    标签: python django decorator lazy-evaluation class-attributes


    【解决方案1】:

    Pyramid 框架有一个非常棒的装饰器,叫做reify,但它只能在实例级别工作,而且你需要类级别,所以让我们稍微修改一下

    class class_reify(object):
        def __init__(self, wrapped):
            self.wrapped = wrapped
            try:
                self.__doc__ = wrapped.__doc__
            except: # pragma: no cover
                pass
    
        # original sets the attributes on the instance
        # def __get__(self, inst, objtype=None):
        #    if inst is None:
        #        return self
        #    val = self.wrapped(inst)
        #    setattr(inst, self.wrapped.__name__, val)
        #    return val
    
        # ignore the instance, and just set them on the class
        # if called on a class, inst is None and objtype is the class
        # if called on an instance, inst is the instance, and objtype 
        # the class
        def __get__(self, inst, objtype=None):
            # ask the value from the wrapped object, giving it
            # our class
            val = self.wrapped(objtype)
    
            # and set the attribute directly to the class, thereby
            # avoiding the descriptor to be called multiple times
            setattr(objtype, self.wrapped.__name__, val)
    
            # and return the calculated value
            return val
    
    class Test(object):
        @class_reify
        def foo(cls):
            print("foo called for class", cls)
            return 42
    
    print(Test.foo)
    print(Test.foo)
    

    运行程序并打印

    foo called for class <class '__main__.Test'>
    42
    42
    

    【讨论】:

    • 像魅力一样工作。非常感谢,我希望我能多次投票!
    猜你喜欢
    • 1970-01-01
    • 2020-06-20
    • 1970-01-01
    • 2019-07-29
    • 1970-01-01
    • 2015-10-23
    • 2017-11-19
    • 2019-08-08
    • 1970-01-01
    相关资源
    最近更新 更多