无法迭代Python中的列表类

时间:2010-04-16 12:10:11

标签: python

我正在尝试使用pyplurk为Plurk编写一个简单的GUI前端。

我已成功创建API连接,登录,检索并显示朋友列表。现在我正在尝试检索并显示Plurks列表。

pyplurk提供了一个GetNewPlurks函数,如下所示:

  def GetNewPlurks(self, since):
    '''Get new plurks since the specified time.
    Args:
      since: [datetime.datetime] the timestamp criterion.
    Returns:
      A PlurkPostList object or None.
    '''
    offset = jsonizer.conv_datetime(since)
    status_code, result = self._CallAPI('/Polling/getPlurks', offset=offset)
    return None if status_code != 200 else \
           PlurkPostList(result['plurks'], result['plurk_users'].values())

如您所见,这会返回一个PlurkPostList,后者又定义如下:

class PlurkPostList:
  '''A list of plurks and the set of users that posted them.'''
  def __init__(self, plurk_json_list, user_json_list=[]):
    self._plurks = [PlurkPost(p) for p in plurk_json_list]
    self._users = [PlurkUser(u) for u in user_json_list]
  def __iter__(self):
    return self._plurks
  def GetUsers(self):
    return self._users
  def __eq__(self, other):
    if other.__class__ != PlurkPostList: return False
    if self._plurks != other._plurks: return False
    if self._users != other._users: return False
    return True

现在我希望能够做到这样的事情:

api = plurk_api_urllib2.PlurkAPI(open('api.key').read().strip(), debug_level=1)
plurkproxy = PlurkProxy(api, json.loads)
user = plurkproxy.Login('my_user', 'my_pass')
ps = plurkproxy.GetNewPlurks(datetime.datetime(2009, 12, 12, 0, 0, 0))
print ps
for p in ps:
  print str(p)

当我运行时,我实际得到的是:

<plurk.PlurkPostList instance at 0x01E8D738>

来自“打印ps”,然后:

    for p in ps:
TypeError: __iter__ returned non-iterator of type 'list'

我不明白 - 肯定列表可迭代的吗?我哪里出错了 - 如何访问PlurkPostList中的Plurks?

3 个答案:

答案 0 :(得分:14)

当您定义自己的__iter__方法时,您应该意识到__iter__方法应返回迭代器,而不是可迭代。您正在返回一个列表,而不是列表的迭代器,因此它失败了。例如,您可以通过执行return iter(self._plurks)来修复它。

如果你想做一些更复杂的事情,比如处理self._plurks中的每个项目,因为它正在被迭代,通常的技巧是让你的__iter__方法成为生成器。这样,对__iter__的调用的返回值是生成器,它是一个迭代器:

def __iter__(self):
    for item in self._plurks:
        yield process(item)

答案 1 :(得分:5)

__iter__方法应该返回一个实现next()方法的对象。

列表没有next()方法,但它有__iter__方法,返回listiterator个对象。 listiterator对象具有next()方法。

你应该写:

def __iter__(self):
    return iter(self._plurks)

答案 2 :(得分:0)

作为替代方案,您还可以定义next()函数并让__iter__()返回self。请参阅Build a Basic Python Iterator以获得一个很好的例子。