PyFunctionObject 在 Py_Finalize 中使程序崩溃,如果它在 PyTuple 中被设置了项目

PyFunctionObject crash the programm in Py_Finalize if it has being set item in PyTuple

本文关键字:PyTuple 设置 项目 如果 崩溃 Py Finalize PyFunctionObject 程序      更新时间:2023-10-16

我需要通过python文件获取函数名称和签名的列表。 Python 函数inspect.signature可以得到我的函数签名。 但是inspect.signature需要一个函数对象(PyFunctionObject(。inspect.getmembers可以返回这样的对象。 但是如果我在PyTuplePyFunctionObject设置为项目,当执行Py_Finalize时 - 我得到错误。 另外,我正在使用Qt。

所以行PyTuple_SetItem(pFuncLink_arg, 0, pGetmember_tupleList);结果错误。

法典:。

PyObject *pName, *pModule;
PyObject *pInspect_module_name, *pInspect_module;
PyObject *pGetmember_function, *pGetmember_call_args, *pGetmembers_reply, *pGetsignature_function, *pGetmember_itemList, *pGetmember_tupleList;
PyObject *pFuncName, *pFuncLink_arg, *pFuncSign;
Py_Initialize();
QFileInfo fileInfo(QFile("X:/Projects/p-text.py"));
QString absPath = fileInfo.absolutePath();
QString fileName = fileInfo.baseName();
PyObject* sysPath = PySys_GetObject((char*)"path"); 
PyObject* programName = PyUnicode_FromString(absPath.toAscii());
PyList_Append(sysPath, programName);
pName = PyUnicode_FromString(fileName.toLatin1());
pModule = PyImport_Import(pName);
Py_DECREF(pName);
if (pModule != NULL){
pInspect_module_name = PyUnicode_DecodeFSDefault("inspect");
pInspect_module = PyImport_Import(pInspect_module_name);
pGetmember_function = PyObject_GetAttrString(pInspect_module, "getmembers");    
pGetsignature_function = PyObject_GetAttrString(pInspect_module, "signature");
pGetmember_call_args = PyTuple_New(1);
PyTuple_SetItem(pGetmember_call_args, 0, pModule);
pGetmembers_reply = PyObject_CallObject(pGetmember_function, pGetmember_call_args);
if (pGetmembers_reply){
Py_ssize_t const num_args = PyList_Size(pGetmembers_reply);
for (Py_ssize_t i = 0; i < num_args; ++i){
pGetmember_itemList = PyList_GetItem(pGetmembers_reply, i);
pGetmember_tupleList =  PyTuple_GetItem(pGetmember_itemList, 1);
if (PyFunction_Check(pGetmember_tupleList)){
pFuncName =  PyTuple_GetItem(pGetmember_itemList, 0);
QString funcName = PyObjectToString(pFuncName);
Py_DECREF(pFuncName);
pFuncLink_arg = PyTuple_New(1);
PyTuple_SetItem(pFuncLink_arg, 0, pGetmember_tupleList);
pFuncSign = PyObject_CallObject(pGetsignature_function, pFuncLink_arg);
if (pFuncSign == NULL) {
if (PyErr_Occurred()) {
PyErr_Print(); 
}
return;
}
QString funcSign = PyObjectToString(pFuncSign);
Py_DECREF(pFuncSign);
cout<<funcSign.toStdString()<<endl;
}
Py_DECREF(pGetmember_tupleList);
Py_DECREF(pGetmember_itemList);
}
}
Py_DECREF(pInspect_module_name);
Py_DECREF(pInspect_module);
Py_DECREF(pGetmember_function);
Py_DECREF(pGetsignature_function);
Py_DECREF(pGetmember_call_args);
}
else {
PyErr_Print();
}
Py_Finalize();

Py_Finalize()引发错误 "project_name.exe触发了断点">

引用计数错误。

PyTuple_GetItem返回"借用的引用"。您不拥有pGetmember_tupleList.

PyTuple_SetItem(在同一链接中找到它..."窃取参考",即假设您在通话前拥有pGetmember_tupleList,但在通话后您不拥有它。

您需要在获取后添加Py_INCREF(pGetmember_tupleList)


您有一个单独的、相关的引用计数错误pFuncName- 您拒绝了您不拥有的借用引用。我怀疑这里可能还有其他引用计数错误......


你也没有错误检查;几乎每个Python调用后面都应该有一个检查,看看它是否引发了异常(通常是NULL指针(。