Как вернуть список целых в расширении Python C API с помощью PyList?
Я строю расширение Python (.pyd
) с использованием Visual Studio 2015 C++ и Python 2.7 32bit.
Это мое .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 ответ
Вы можете изменить формат 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_New
а также PyList_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
указатель приведен к целому числу.