我正在搞乱,试图建立类似于IPython / Jupyter笔记本的东西。我在QT5中编写我的应用程序,其中很多都与在本机应用程序中“嵌入”Python有关。
我想出了如何嵌入python以及如何允许它执行用户输入的脚本。我希望能够使用绘图库(例如matplotlib),并在我的应用程序中显示它们的输出。 (事实上,我试图做的事情似乎与this question中描述的非常相似。)
然而,当我尝试使用import matplotlib.pyplot
导入绘图库时,我的应用程序会出现段错误(我试过调试,但崩溃不在我的代码中,所以我无法从中得到任何明智的结果)。
我用来初始化嵌入式Python并运行任意脚本的代码显示在这个问题的底部。
我可以导入其他库(例如sys
或numpy
)。我可以导入matplotlib
罚款。但是当我尝试导入matplotlib.pyplot
时,会出现段错误。
有人有任何建议吗?
编辑: 我已经确定原因在于(出于某种原因)我使用QT。当我编译一个导入matplotlib的简单C或C ++程序时,它不段错误...
我的代码:
#include "pythoninteractor.h"
#include <QString>
#include <Python.h>
#include <string>
#include <QList>
PythonInteractor::PythonInteractor()
{
this->pyOutput_redir =
"import sys\n\
class CatchOutErr:\n\
def __init__(self):\n\
self.value = ''\n\
def write(self, txt):\n\
self.value += txt\n\
catchOutErr = CatchOutErr()\n\
sys.stdout = catchOutErr\n\
sys.stderr = catchOutErr\n\
"; //this is python code to redirect stdouts/stderr
QString paths[] = {"",
"/home/tcpie/anaconda3/lib/python35.zip",
"/home/tcpie/anaconda3/lib/python3.5",
"/home/tcpie/anaconda3/lib/python3.5/plat-linux",
"/home/tcpie/anaconda3/lib/python3.5/lib-dynload",
"/home/tcpie/anaconda3/lib/python3.5/site-packages",};
Py_SetProgramName(L"qt-notepad-tut");
Py_Initialize();
PyObject *pModule = PyImport_AddModule("__main__"); //create main module
PyRun_SimpleString(this->pyOutput_redir.toStdString().c_str()); //invoke code to redirect
PyObject *sys_path;
PyObject *path;
sys_path = PySys_GetObject("path");
if (sys_path == NULL)
return;
PySequence_DelSlice(sys_path, 0, PySequence_Length(sys_path));
for (size_t i = 0; i < sizeof(paths) / sizeof(QString); i++) {
path = PyUnicode_FromString(paths[i].toStdString().c_str());
if (path == NULL)
continue;
if (PyList_Append(sys_path, path) < 0)
continue;
}
}
QString PythonInteractor::run_script(QString script)
{
QString ret = "";
PyObject *pModule = PyImport_AddModule("__main__");
PyRun_SimpleString(script.toStdString().c_str());
PyErr_Print(); //make python print any errors
PyObject *catcher = PyObject_GetAttrString(pModule,"catchOutErr"); //get our catchOutErr created above
if (catcher == NULL) {
Py_Finalize();
return ret;
}
PyObject *output = PyObject_GetAttrString(catcher,"value"); //get the stdout and stderr from our catchOutErr object
if (output == NULL) {
return ret;
}
ret = QString(PyUnicode_AsUTF8(output));
return ret;
}
答案 0 :(得分:2)
此次崩溃的原因是QT版本之间存在冲突。
首先,可以使用以下最小代码重现该问题。评论&#34; Q_OBJECT&#34; main.h中的行可以防止崩溃。
文件main.h:
#ifndef MAIN_H
#define MAIN_H
#include <QMainWindow>
class test : public QMainWindow
{
Q_OBJECT // Commenting out this line prevents the crash
};
#endif // MAIN_H
文件main.cpp:
#include <Python.h>
#include "main.h"
int main()
{
Py_Initialize();
PyRun_SimpleString("import matplotlib.pyplot as plt");
PyRun_SimpleString("print('If we are here, we did not crash')");
Py_Finalize();
return 0;
}
我正在通过Anaconda运行Python3。但是,我已经通过我的包管理器安装了QT5(在我的例子中:在Ubuntu上使用apt-get)。我怀疑问题在于我的Anaconda安装的matplotlib使用的是与我通过我的包管理器安装的版本不同的QT5版本。
修复很简单:通过我的包管理器安装matplotlib可以解决问题! (在我的Ubuntu系统上:sudo apt-get install python3-matplotlib
)