python中如何实现__getitem__ dunder方法获取属性值?
Posted
技术标签:
【中文标题】python中如何实现__getitem__ dunder方法获取属性值?【英文标题】:How to implement the __getitem__ dunder method in python to get attribute values? 【发布时间】:2019-01-15 01:15:43 【问题描述】:我有以下代码:
class Personne:
def __init__(self, name, age):
self.name = name
self.age = age
def __getitem__(self, *args):
keys = list(*args)
return [self.__dict__[key] for key in keys]
if __name__ == '__main__':
p = Personne('A', 20)
# age = p['age']
# print(age)
# name = p['name']
# print(name)
name, age = p['name', 'age']
print(name, age)
未注释的部分可以正常工作,但是注释代码中存在问题。 如何实现所需的行为,即根据传递给 getitem 方法的参数(可以是一个或多个)获取属性值。
谢谢。
【问题讨论】:
【参考方案1】:我相信您知道,目前您的代码返回错误,因为 keys = list(*args)
将字符串 age
视为可迭代并尝试在其中查找 a
、g
和self.__dict__
。因此,您可以检查提供给__getitem__
的参数的类型,并对args
执行适当的转换以返回keys
:
class Personne:
def __init__(self, name, age):
self.name = name
self.age = age
def __getitem__(self, args):
if isinstance(args, str):
keys = [ args ] # args is a string
else:
keys = list(args) # args is a tuple
return [self.__dict__[key] for key in keys]
输出将是来自self.__dict__
的值列表。
【讨论】:
您的代码每次都返回一个列表,即使是一个输出也是如此。但是@Levi Levi 为我解决了这个问题。谢谢。 是的,它会这样做——请参阅返回声明:return [self.__dict__[key] for key in keys]
。我倾向于更喜欢函数每次都返回相同的数据类型,这样我就可以使用相同的代码来处理返回,而无需进行另一次类型检查。更清楚地指定您的输入、问题和预期输出,人们可以更有效地提供帮助。【参考方案2】:
只传递一个字符串的问题在于,当你调用 p['age'] 时,*args 的类型是字符串,因此 list(*args) 变为 ['a', 'g', ' e']。当你同时传递 'age' 和 'name' 时,*args 被解释为一个元组,你会得到适当的结果。处理此问题的一种方法是检查 getitem(self, *args) 方法中 *args 参数的类型。以下修改将解决您面临的问题(Python 3),但您可能需要稍微清理一下。
class Personne:
def __init__(self, name, age):
self.name = name
self.age = age
def __getitem__(self, *args):
if isinstance(*args, str):
return self.__dict__[str(*args)]
keys = list(*args)
return [self.__dict__[key] for key in keys]
if __name__ == '__main__':
p = Personne('A', 20)
age = p['age']
print(age)
name = p['name']
print(name)
name, age = p['name', 'age']
print(name, age)
CONSOLE OUTPUT :
20
A
A 20
【讨论】:
以上是关于python中如何实现__getitem__ dunder方法获取属性值?的主要内容,如果未能解决你的问题,请参考以下文章
如何让 Python 2 的 __getitem__ 在类上工作? [复制]
python 魔法方法之:__getitem__ __setitem__ __delitem__
python 的__len__,__getitem__ __setitem__ __delitem__ __contains__