gpt4 book ai didi

python - 惰性类属性装饰器

转载 作者:行者123 更新时间:2023-11-28 20:47:43 25 4
gpt4 key购买 nike

我有一个 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 装饰器,但没有一个同时适用于这两个装饰器,而且我不知道如何自己编写一个。

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

最佳答案

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

关于python - 惰性类属性装饰器,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/18289871/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com