我不知道我想做的是不是太不象话了,我只是想做错事,或者我不知道如何正确地问这个问题。能做到这一点对我来说很有意义,但我已经找了15种不同的方法,却找不到答案。
我想做的似乎很简单:我有一个对象列表。我想通过对象的属性访问该列表。此代码有效:class Fruit:
def __init__(self, name, color):
self.name = name
self.color = color
def __str__(self):
return self.name
class BaseballPlayer:
def __init__(self, name, number, position):
self.name = name
self.number = number
self.position = position
def __str__(self):
return self.name
class ListByProperty(list):
def __init__(self, property, list):
super(ListByProperty, self).__init__(list)
self.property = property
def __getitem__(self, key):
return [item for item in self if getattr(item, self.property) == key][0]
fruits = ListByProperty('name', [Fruit('apple', 'red'), Fruit('banana', 'yellow')])
baseballTeam = ListByProperty('position', [BaseballPlayer('Greg Maddux', 31, 'P'),
BaseballPlayer('Javy Lopez', 8, 'C')])
teamByNumber = ListByProperty('number', baseballTeam)
print 'Apples are', fruits['apple'].color
pitcher = baseballTeam['P']
print 'The pitcher is #%s, %s' % (pitcher.number, pitcher)
print '#8 is', teamByNumber[8]
>>> Apples are red
The pitcher is #31, Greg Maddux
#8 is Javy Lopez
但是,我真的要让我自己的列表类做这么简单的事情吗?除了循环或listcomp,没有其他通用方法吗?这看起来应该是一件非常常见的事情,通过对象的属性在列表中拥有对象和访问项的列表。似乎应该以类似于sorted(key=...)的方式来支持它。
请注意,这与需要dict的情况不同。事实上,使用对象列表而不是dict的整个目的是避免执行以下操作:fruits = {'apple': Fruit('apple', 'red')}
…这需要您键入apple两次。似乎应该有一种通用的方法来做这样的事情:print 'Apples are', fruits['apple'].color
……不需要对list进行子类划分。
好吧,你可以这样做:fruits = [Fruit('apple', 'red'), Fruit('banana', 'yellow')]
fruits = {f.name: f for f in fruits}
或者你可以把它划一行,但看起来…呃…语法上还是很酸的?:)
到目前为止,我发现最好的办法是:class DictByProperty(dict):
def __init__(self, property, list):
super(DictByProperty, self).__init__({getattr(i, property): i for i in list})
self.property = property
fruits = DictByProperty('name', [Fruit('apple', 'red')])
哦,好吧,谢谢,我已经从这个问题中学到了很多东西。