13

为什么以下 Python 代码会引发错误
TypeError: type object argument after * must be a sequence, not generator
,而如果我在生成器 f 中注释第一行(无用),一切正常?

from itertools import izip

def z():
    for _ in range(10): 
        yield _

def f(z):
    for _ in z: pass    # if I comment this line it works! (??)
    for x in range(10):
        yield (x,10*x,100*x,1000*x)

iterators =  izip(*f(z))
for it in iterators:
    print list(it)

注意我实际上想要做的是,使用单个生成器返回多个迭代器(我将作为参数传递给生成器的数量)。我发现这样做的唯一方法是产生元组并在它们上使用 izip() - 对我来说是黑魔法。

4

1 回答 1

27

这很有趣:z当你把它传递给时你忘了打电话f

iterators =  izip(*f(z()))

所以f试图迭代一个函数对象:

for _ in z: pass  # z is a function

这引发了 TypeError:

TypeError: 'function' object is not iterable

Python 内部人员抓住了它并重新提出了一条令人困惑的错误消息。

# ceval.c

static PyObject *
ext_do_call(PyObject *func, PyObject ***pp_stack, int flags, int na, int nk)
{
 ...

            t = PySequence_Tuple(stararg);
            if (t == NULL) {
                if (PyErr_ExceptionMatches(PyExc_TypeError)) {
                    PyErr_Format(PyExc_TypeError,
                                 "%.200s%.200s argument after * "
                                 "must be a sequence, not %200s",
                                 PyEval_GetFuncName(func),
                                 PyEval_GetFuncDesc(func),
                                 stararg->ob_type->tp_name);
 ...
于 2013-03-21T23:10:19.030 回答