谢谢你的帮助——这个问题已经被问过很多次了,但是我没有找到一个完整的答案。我将嵌入的Python 3.4.2添加到使用MS MFC类编写的C++中的现有模拟器工具中。应用程序是多线程的,因此用户可以执行Python脚本并与模拟器系统交互。在
如何成功退出? 我是否按正确的顺序使用GIL和thread state命令? 我是否过早地结束了Python解释器线程并破坏了Python线程合并机制?在
我的问题是,当我调用Py_Finalize时,它调用wait_for_thread_shutdown,然后调用PyThreadState_Get,并遇到一个致命错误“PyThreadState_Get:no current thread”。基于检测到致命错误的点,它似乎与多线程嵌入式Python应用程序末尾的线程合并有关。在
我压缩了我的代码,以澄清它,并消除任何似乎不相关的东西。如果我走得太远或不够远,我很抱歉。主线程初始化并完成Python。在
BOOL CSimApp::InitInstance()
{
...
// Initialize command table for appl. object and for documents
int iReturn = PyImport_appendInittab("sim", &PyInit_SimApp);
iReturn = PyImport_AppendInittab("sim_doc", &PyInit_SimDoc);
// Initialize Python and prepar to create threads
_pHInstance = new CPyInstance();
...
}
int CSimApp::ExitInstance()
{
...
if (_pHInstance) {
delete _pHInstance;
_pHInstance = NULL;
}
...
}
我使用实用程序类创建Python实例(CPyInstance)并管理pythongil(ACQUIRE_PY_GIL)。初始化应用程序时,还会创建CPyInstance的实例。CPyInstance类初始化并完成Python线程管理。Python全局锁管理是通过ACQUIRE_PY_GIL和RELEASE_PY_GIL结构来实现的。在
^{pr2}$Python解释器线程是为了响应CMainFrame窗口处理的Windows消息而创建的。Python线程和解释器响应用户命令运行。当用户完成解释器(Control-Z)时,解释器退出,线程清除并删除Python线程状态,然后线程自行终止。在
void CMainFrame::OnOpenPythonInterpreter()
{
// Create PyThread thread
m_pPyThread = (CPyThread*)AfxBeginThread(RUNTIME_CLASS(CPyThread),
THREAD_PRIORITY_BELOW_NORMAL,0, CREATE_SUSPENDED);
CMainFrame* mf = (CMainFrame*)theApp.m_pMainWnd;
m_pPyThread->SetOwner(this,((CWnd*)mf)->GetSafeHwnd());
m_pPyThread->CreateLocks(&m_PyThreadEvent,&m_PyThreadBusyMutex);
m_pPyThread->ResumeThread();
}
CPyThread类实际上调用Python解释器。当解释器返回时,GIL被释放,Python线程状态被清除和删除。线程终止以响应PostQuitMessage。在
int CPyThread::Run()
{
PyEval_AcquireLock();
PyInterpreterState * mainInterpreterState = CPyInstance::mainThreadState->interp;
PyThreadState * myThreadState = PyThreadState_New(mainInterpreterState);
PyEval_ReleaseLock();
try {
ACQUIRE_PY_GIL lock;
FILE* fp1 = stdin;
char *filename = "Embedded";
PyRun_InteractiveLoop(fp1, filename);
} catch(const std::exception &e) {
safe_cout << "Exception in PyRun_InteractiveLoop: " << e.what() << "\n";
} catch(...) {
std::cout << "Exception in Python code: UNKNOWN\n";
}
PyThreadState_Clear(myThreadState);
PyThreadState_Delete(myThreadState);
::PostQuitMessage(0);
return 0;
}
int CPyThread::ExitInstance()
{
return CWinThread::ExitInstance();
}
在“user4815162342”的建议下,我修改了~CPyInstance()析构函数,在调用Py_Finalize()之前获取GIL。现在我的应用程序可以正常退出了,谢谢。在
inline CPyInstance::~CPyInstance()
{
try {
PyGILState_STATE state = PyGILState_Ensure();
Py_Finalize();
} catch(const std::exception &e) {
safe_cout << "Exception in ~CPyInstance(): " << e.what() << "\n";
} catch(...) {
std::cout << "Exception in Python code: UNKNOWN\n";
}
}
您正在调用
Py_Finalize
,而没有保持全局解释器锁。这是不允许的:除了获取GIL本身的调用之外,每个pythonapi调用都必须持有锁。在ACQUIRE_PY_GIL
RAII guard在这个目的上没有用处,因为它会在Py_Finalize
返回后释放GIL——在这种情况下,必须在没有匹配的释放的情况下调用PyGILState_Ensure
。在相关问题 更多 >
编程相关推荐