如何使用PyList返回Python C API扩展中的int列表?

时间:2018-06-03 17:13:27

标签: python c++ cpython python-c-api python-extensions

我正在使用Visual Studio 2015 C ++项目和Python 2.7 32位构建Python扩展(.pyd)。

这是我的.cpp文件:

#include <Python.h>

static PyObject* GetTwoInts(PyObject* self, PyObject* args)
{
    srand(time(NULL));
    int random1 = rand() % 10;
    int random2 = rand() % 10;

    PyObject * python_val = Py_BuildValue("ii", random1, random2);
    return python_val;
}

static PyObject* GetListTwoInts(PyObject* self, PyObject* args)
{
    srand(time(NULL));
    int random1 = rand() % 10;
    int random2 = rand() % 10;

    PyObject *val1 = PyInt_FromLong(random1);
    PyObject *val2 = PyInt_FromLong(random2);

    PyObject *result = PyList_New(2);
    PyList_SetItem(result, 0, val1);
    PyList_SetItem(result, 1, val2);

    PyObject * python_val = Py_BuildValue("ii", result);
    return python_val;
}

PyMODINIT_FUNC initUtils(void)
{
    static PyMethodDef methods[] = {
        { "GetTwoInts", GetTwoInts, METH_NOARGS,
        "Get two C ints as a Python tuple with two random numbers" },

        { "GetListTwoInts", GetListTwoInts, METH_NOARGS,
        "Get a list with two random numbers" },
        { NULL, NULL, 0, NULL },

    };

    PyObject *m = Py_InitModule("Utils", methods);
}

这是使用编译扩展名的Python源代码:

import sys
import Utils
print help(Utils)
print Utils.GetTwoInts()
print Utils.GetListTwoInts()

这是输出:

(4, 2)
(91213912, 91213912)

所以,Py_BuildValue("ii", random1, random2);给了我一个带有两个随机整数的正确元组,正如预期的那样。但是,在GetListTwoInts方法中返回一个列表会给出无效数字(看起来像是引用值还是指针?)。

如何在GetListTwoInts方法中返回实际值列表?

1 个答案:

答案 0 :(得分:5)

您可以更改Py_BuildValue的格式,以便构建列表而不是元组。只需使用"[ii]"代替"ii"作为第一个参数:

static PyObject* GetListTwoInts(PyObject* self, PyObject* args)
{
    srand(time(NULL));
    int random1 = rand() % 10;
    int random2 = rand() % 10;

    PyObject * python_val = Py_BuildValue("[ii]", random1, random2);
    return python_val;
}

如果要创建动态尺寸列表,可以使用PyList_NewPyList_SetItem

static PyObject* GetList(PyObject* self, PyObject* args)
{
    srand(time(NULL));
    int const N = 10;
    PyObject* python_val = PyList_New(N);
    for (int i = 0; i < N; ++i)
    {
        int r = rand() % 10;
        PyObject* python_int = Py_BuildValue("i", r);
        PyList_SetItem(python_val, i, python_int);
    }
    return python_val;
}

问题中PyList版本的问题在于您在列表中使用Py_BuildValue("ii", result)。这会尝试创建一个包含两个整数的元组,其中第一个值是result指针转换为整数。