如何在Python中处理来自C ++的PyObject *

时间:2013-11-29 08:34:25

标签: c++ python dll ctypes pyobject

我创建了用C ++编写的DLL,导出函数返回PyObject *。然后我使用ctypes在Python中导入DLL。现在,我怎样才能得到真正的PyObject ??

这是c ++代码的一部分:

PyObject* _stdcall getList(){

    PyObject * PList = NULL;
    PyObject * PItem = NULL;
    PList = PyList_New(10);

    vector <int> intVector;
    int i;
    for(int i=0;i<10;i++){
        intVector.push_back(i);
    }

    for(vector<int>::const_iterator it=intVector.begin();it<intVector.end();it++){
        PItem = Py_BuildValue("i", &it);
        PyList_Append(PList, PItem);
    }
    return PList;
}

和一些python代码:

dll = ctypes.windll.LoadLibrary(DllPath)
PList = dll.getList()

* 我想获得包含1,2,3,4 ... 10的真正的python列表? * 我清楚吗?谢谢提前

3 个答案:

答案 0 :(得分:5)

您的代码有很多问题,有些修改:

#include <Python.h>
#include <vector>

extern "C" PyObject* _stdcall getList(){
  PyObject *PList = PyList_New(0);

  std::vector <int> intVector;
  std::vector<int>::const_iterator it;

  for(int i = 0 ; i < 10 ; i++){
    intVector.push_back(i);
  }

  for(it = intVector.begin(); it != intVector.end() ; it++ ){
    PyList_Append(PList, Py_BuildValue("i", *it));
  }

  return PList;
}

编译它:

> g++ -Wall -shared lib.cpp -I \Python27\include -L \Python27\libs -lpython27 -o lib.dll -Wl,--add-stdcall-alias

现在您可以将其作为任何函数加载,并将getList返回类型设置为py_object

import ctypes

lib = ctypes.WinDLL('lib.dll')

getList = lib.getList
getList.argtypes = None
getList.restype = ctypes.py_object

getList()

测试它:

>>> import ctypes
>>>
>>> lib = ctypes.WinDLL('lib.dll')
>>>
>>> getList = lib.getList
>>> getList.argtypes = None
>>> getList.restype = ctypes.py_object
>>> getList()
[0, 1, 2, 3, 4, 5, 6, 7, 8, 9]
>>>
>>>

答案 1 :(得分:1)

使用Visual Studio和Python 64位:
1-创建一个空的Win32项目(DLL类型)
2-右键单击解决方案项目 - &gt;配置管理器
3- Active Solution配置(Release)
4-有源解决方案平台 - &gt;新建,然后在底部的下拉列表中,选择x64 - &gt; OK
5-在“源文件”文件夹中,添加一个空的C ++文件
6-输入您的C ++代码(可以识别getList的一个修改)

#include <Python.h>
#include <vector>

extern "C" __declspec(dllexport) PyObject* _stdcall getList();

PyObject* _stdcall getList(){


    PyObject *PList = PyList_New(0);

    std::vector <int> intVector;
    std::vector<int>::const_iterator it;

    for (int i = 0; i < 10; i++){
        intVector.push_back(i);
    }

    for (it = intVector.begin(); it != intVector.end(); it++){
        PyList_Append(PList, Py_BuildValue("i", *it));
    }

    return PList;
}

答案 2 :(得分:0)

我不清楚你在问什么。但我想你的意思是问你现在可以用你的DLL做什么。

  1. 好吧,为了正确使用它,你必须构建一个特殊的DLL,它可以直接作为Python中的模块导入。为了确定如何使用它,最好先查看其他模块,以及它们是如何做到的。 E. g。 MySQLdb可能是候选人。

    简而言之,你有这个“包装器”DLL调用你的函数。

  2. 但如果我现在再看一下你的问题,我发现你正试图通过ctypes加载你的DLL。这也是可行的,甚至可能更好,你必须使用ctypes.py_object data type