来自Python 3 async for loop的TypeError

时间:2017-03-06 21:21:00

标签: python asynchronous python-3.5

我正在学习Python相对较新的异步功能。我在PEP 492中找到了这个:

  

以下是将常规迭代转换为的实用程序类   异步的。虽然这不是一件非常有用的事情,但是   代码说明了常规和异步之间的关系   迭代器。

class AsyncIteratorWrapper:
    def __init__(self, obj):
        self._it = iter(obj)

    def __aiter__(self):
        return self

    async def __anext__(self):
        try:
            value = next(self._it)
        except StopIteration:
            raise StopAsyncIteration
        return value

async for letter in AsyncIteratorWrapper("abc"):
    print(letter)

我试图通过向函数添加给定的async for循环来运行此代码,然后使用事件循环调用它。

完整的示例代码(在解释器中运行):

class AsyncIteratorWrapper:
    def __init__(self, obj):
        self._it = iter(obj)
    def __aiter__(self):
        return self
    async def __anext__(self):
        try:
            value = next(self._it)
        except StopIteration:
            raise StopAsyncIteration
        return value

async def aprint(str):
  async for letter in AsyncIteratorWrapper(str):
    print(letter)

import asyncio
loop = asyncio.get_event_loop()
co = aprint("abcde")
loop.run_until_complete(co)

然而,我收到一个错误:

Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
  File "/opt/rh/rh-python35/root/usr/lib64/python3.5/asyncio/base_events.py", line 337, in run_until_complete
    return future.result()
  File "/opt/rh/rh-python35/root/usr/lib64/python3.5/asyncio/futures.py", line 274, in result
    raise self._exception
  File "/opt/rh/rh-python35/root/usr/lib64/python3.5/asyncio/tasks.py", line 239, in _step
    result = coro.send(None)
  File "<stdin>", line 2, in aprint
TypeError: 'async for' received an invalid object from __aiter__: AsyncIteratorWrapper

我做错了什么?如何修复这个例子?我对PEP的代码失败感到有些惊讶。

我正在使用python版本3.5.1。

1 个答案:

答案 0 :(得分:0)

您使用的代码适用于python 3.5.2 +。

从Python 3.5.2 __aiter__可以直接返回异步迭代器。 More here

您收到的错误是因为较旧的python(3.5.1),因此返回了错误的类型。