我正在尝试在使用CMake编译的C ++项目中包含python文件。
首先,我使用这两个文件独立完成此操作:
#include <Python.h>
#include <stdio.h>
int main(int argc, char *argv[])
{
setenv("PYTHONPATH", ".", 0);
char hostname[] = "localhost";
PyObject *pName, *pModule, *pFunc;
PyObject *pArgs, *pValue;
Py_Initialize();
pName = PyString_FromString("GetHostname");
pModule = PyImport_Import(pName);
Py_DECREF(pName);
if(pModule != NULL) {
pFunc = PyObject_GetAttrString(pModule, "GetHostname");
if (pFunc && PyCallable_Check(pFunc)) {
pArgs = PyTuple_New(1);
pValue = PyString_FromString(hostname);
PyTuple_SetItem(pArgs, 0, pValue);
pValue = PyObject_CallObject(pFunc, pArgs);
Py_DECREF(pArgs);
if (pValue != NULL) {
printf("The IP address is %s\n", PyString_AsString(pValue));
Py_DECREF(pValue);
}
else {
Py_DECREF(pFunc);
Py_DECREF(pModule);
PyErr_Print();
fprintf(stderr, "Call Failed\n");
return 1;
}
}
else {
if (PyErr_Occurred())
PyErr_Print();
fprintf(stderr, "Cannot find function\n");
}
Py_XDECREF(pFunc);
Py_DECREF(pModule);
}
else {
PyErr_Print();
fprintf(stderr, "Failed to load file\n");
return 1;
}
Py_Finalize();
return 0;
}
和
import socket
def GetHostname(hostname):
addr = socket.gethostbyname(hostname)
return addr
当我使用
编译时g++ $(python-config --cflags) -o test $(python-config --ldflags) ./test.cpp
来自how to link python static library with my c++ program一切正常。
但这包含在使用CMake编译的项目中,我必须做错了,因为在编译后我得到了
Traceback (most recent call last):
File "/src/GetHostname.py", line 1, in <module>
import socket
File "/usr/lib64/python2.6/socket.py", line 46, in <module>
import _socket
ImportError: /usr/lib64/python2.6/lib-dynload/_socketmodule.so: undefined symbol: PyExc_ValueError
在CMakeLists.txt中我添加了行
find_package( PythonInterp REQUIRED )
find_package( PythonLibs REQUIRED )
include_directories ( ${PYTHON_INCLUDE_DIRS} )
add_library (GetHostname MODULE GetHostname.cc)
target_link_libraries(GetHostname ${PYTHON_LIBRARIES})
CONFIGURE_FILE(${PATH_TO_SOURCE}GetHostname.py ${PATH_TO_BUILD}GetHostname.py COPYONLY)
基于此帖子Python.h: No such file or directory
所有内容都编译但由于错误导致python模块无法加载。我没有在CMake中正确链接python库吗?
任何可以解释失败原因的想法都是受欢迎的。
使用Python 2.6
我知道这可以在C ++中完成,但是这不是我需要包含的唯一python模块,所以用C ++重写它不是我正在寻找的答案。我现在也知道localhost的IP地址是已知的,这仅用于测试目的。
答案 0 :(得分:2)
所以我终于找到了答案。
问题是当在CMake中链接时,与python库的链接只在本地加载(RTLD_LOCAL),这意味着实际的python脚本没有与python库链接,只有C ++。
要解决此问题,只需在C ++代码中加载具有全局符号解析功能的python库。
dlopen("libpython2.6.so.1.0", RTLD_NOW | RTLD_NOLOAD | RTLD_GLOBAL);
这是一个快速而又脏的修复,因为它特定于python2.6,你应该通过在CMake中根据机器上安装的版本定义变量来使其可移植。
您必须导入dlfcn才能使用dlopen:
#include <dlfcn.h>