gpt4 book ai didi

python - 如何从描述符将属性请求委托(delegate)给 MRO 链

转载 作者:行者123 更新时间:2023-12-01 04:46:22 25 4
gpt4 key购买 nike

如果我有一个父类和一个子类,当请求属性时,我会在 Python 中得到以下行为:

class Parent():
i = 1

class Child(Parent):
def __init__(self, j = 2):
self.j = j

ch = Child()
print(ch.i, ch.j) # 1 2

ij属性的请求按预期沿着MRO链向上; i 在父类属性中找到,j 在实例属性中找到。

现在,如果我添加通用描述符并替换 Child 中的 j 属性,就会发生这种情况:

class _Attr():
def __init__(self, attr_name):
self.attr_name = '_' + attr_name
def __get__(self, instance, klass):
return getattr(instance, self.attr_name)

class Child(Parent):
j = _Attr('j')
def __init__(self, j = 2):
self._j = j

ch = Child()
print(ch.i, ch.j) # 1 2

到目前为止,一切都很好。

但是,使用上面的描述符,如果我们这样做:

class Child(Parent):
j = _Attr('j')
i = _Attr('i')
def __init__(self, j = 2):
self._j = j

ch = Child()
print(ch.i, ch.j) # AttributeError: 'Ch" object has no attribute '_i'

由于属性查找失败而发生此错误:

return getattr(ch, '_i')

我想要的是描述符“默默地失败”,并且属性查找继续沿 MRO 链向上进行。我不确定该怎么做。

我试过这个:

class _Attr():
def __init__(self, attr_name):
self.attr_name = '_' + attr_name
def __get__(self, instance, klass):
result = getattr(instance, self.attr_name, None)
if result == None:
return NotImplemented
else:
return result

但这并不能解决问题。我怎样才能得到我想要的行为?我有一种感觉,在这种情况下我需要以某种方式使用 super() ,但我不知道该怎么办。

最佳答案

有两件事。

  1. 您需要将对实际属性名称的引用存储在 _Attr 中,以便您可以在父查找中使用它。

  2. 在查找过程中,您可以将属性获取工作委托(delegate)给Parent类,使用super(klass, instance)

所以你的_Attr看起来像这样

class _Attr():

def __init__(self, attr_name):
self._original_name = attr_name
self.attr_name = '_' + attr_name

def __get__(self, instance, klass):
if hasattr(instance, self.attr_name):
return getattr(instance, self.attr_name)
return getattr(super(klass, instance), self._original_name)

关于python - 如何从描述符将属性请求委托(delegate)给 MRO 链,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/29292254/

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