QT与Python混合编程经验记录

一、如何embedding python,Python文档中有专门一章阐述https://docs.python.org/3.5/extending/embedding.htm;html

一、库文件:在vs--c/c++--附加包含文件中添加目;连接中也要添加,:将python中的include,libs二个目录添加进来python

二、对于Python,没有调试库,可直接将运行库复制一份,名称后面加上_d,就可用,可不能源代码调试而已,同时修改c++

pyconfig.hapi

#ifdef _DEBUG 
//# define Py_DEBUG    //将这行注释掉
#endif 
app

和object.h函数

#if defined(Py_DEBUG) && !defined(Py_TRACE_REFS) 
// #define Py_TRACE_REFS   //将这行注释掉
#endif
ui

三、因为QT中定义了slots做为关键了,而python3中有使用slot做为变量,因此有冲突,办法是:spa

  A、在vs中-属性-c/c++ -预处理器-预处理器定义-中添加:QT_NO_KEYWORDS指令,调试

    在Dt的项目文件中.pro文件中添加 config +=_no_keywords指令,再生成vs的项目文件,也就自动添加了上面的指令orm

    这种方法是全局的,同时也取消 forever,foreach的宏定义,由于不少代码使用foreach,因此反作用大;

  B、在使用第三方代码时暂时取消冲突的宏定义,而后从新定义相关宏,如QT调用第三方python时,只是在python的object.h中 slots冲突,所以修改object.h:

   typedef struct{     

      const char* name;     

      int basicsize;    

       int itemsize;     

      unsigned int flags;

      #undef slots     //这里取消slots宏定义

       PyType_Slot *slots;  /* terminated by slot==0. */

      #define slots Q_SLOTS  //这里恢复slots宏定义与QT中QObjectDefs.h中一致

  } PyType_Spec;  

四、在主程序中执行一开始Py_Initialize(); 时就出现错误

   解决办法是:Py_SetPythonHome((wchar_t*)(L"C:\\Python3"));
是由于我安装python的时候没有添加PYTHONHOME这个环境变量
在Py_Initialize()以前调用下Py_SetPythonHome("C:\\Python3");就能够了 

五、用于导入sys模块,以供程序使用,这是基本模块
PyRun_SimpleString("import sys");

六、在执行pModule = PyImport_ImportModule("jptest"); 语句导入自定义模块时,返回对pModule为NULL,是因为自定义模块的位置不在python系统搜索的系统目录列表内,所以解决办法是在系统搜索目录列表中添加目录:

PyRun_SimpleString("sys.path.append('D:\\develop\\qtpythontest\\Win32\\Debug')");

注意字符中中"\"必须前面加上“\\",也就是单个"\"是转意符;并且目录尾部不能添加多余的\符;若是不能确承认将它们显示出来用下面这个语句:PyRun_SimpleString("print(sys.path)");

七、数据转换,在c/c++与python交互时,都是经过PyObject来传入和传出数据的,Python提供相关函数对PyObject数据进行转换,转换时使用格式字符串来控制生成的对象类型,具体可参见https://docs.python.org/3.5/c-api/arg.html官方文档:

  A) 将c/c++数据转换成PyObject:

    PyObject *pInt=Py_BuildValue("i",2003);

    PyObject *pStr=Py_BuildValue("s","This is a string");

    PyObject *pTuples=Py_BuildValue("()"); //生成空元组,可做为调用不包含任何参数的函数时,传递空参数

    PyObject *pTuples=Py_BuildValue("(s)","This is a string");  //生成一个元素的元组,可做为调用只包含一个字符参数的函数时,传递一个字符参数

  B) 将PyObject数据转换成c/c++数据:

    1) int bb=0;  PyArg_Parse(pObjcet,"i",&bb);   //这里pObject是包含整数数据的Python对象,第二个字符串参数"i"指定转换类型,第三个参数将结果值存入bb变量;

    2) char * cc=NULL;  PyArg_parse(pObject,"s",&cc);   //这是字符串转换

    3) char * cc=NULL;  PyArg_parse(pObject,"(s)",&cc);   //这是包含一个字符串元素元组转换

 

八、调用Python模块函数时,传入参数时,要构造一个参数元组,如:presult = PyObject_CallObject(pfunction, args);这里args就是一个元组,做为被调用函数的参数列表;

  A、如参数为空,则这样构造:args=Py_BuildValues("(si)","abc",10);  表示构造二个参数的元组,一个是字符型,另外一个是整;多个参数,可参照处;

  B、若是参数为空,则需构造一个包含0个元素元组:args=Py_BuildValues("()"); 

  注意以上二种都在格式字符串中包含"()",这是指示构造元;做为函数调用参数必须传递元组,也必须这样构;

  下例是经过可变参数来构造调用函数参数元组:

int PythonHandler::PyModuleRunFunction(const char *module, const char *function,
        const char *result_format, void *result, const char *args_format, ...) {
......
    //这里构造调用函数所使用的参数元组

    va_list args_list; va_start(args_list, args_format); args = Py_VaBuildValue(const_cast<char *>(args_format), args_list); va_end(args_list);
  ...
  if (!args)
	{
        //args为空,则元组构造失败 Py_DECREF(pfunction); return -3; } ... presult = PyObject_CallObject(pfunction,args); //调用函数
相关文章
相关标签/搜索