17

C++ から Python クラスのメソッドを呼び出そうとしています。これが呼び出される C++ メソッドは C++ コールバックです。

このメソッド内で python メソッドを呼び出そうとすると、segmentation fault.

次のようなグローバル変数にPython関数のインスタンスを保存しました

// (pFunc is global variable of type PyObject*)
pFunc = PyDict_GetItemString(pDict, "PlxMsgWrapper");

wherePlxMsgWrapperは、コールバックで使用される Python メソッドです。

コールバックでは、引数は次のように作成されます。

PyObject* args = PyTuple_Pack(2, PyString_FromString(header.c_str()),
                                 PyString_FromString(payload.c_str()));

を作成するとき

PyObject * pInstance = PyObject_CallObject(pFunc, args);

この行では、セグメンテーション違反が発生しています。この後、実際の python メソッドは次のように呼び出されます。

PyObject* recv_msg_func = PyObject_GetAttrString(module, (char *)"recvCallback");
args = PyTuple_Pack(1, pInstance);
PyObject_CallObject(recv_msg_func, args);
4

3 に答える 3

34

C/C++ コールバックから Python 関数を呼び出す場合は、いくつかの作業を行う必要があります。まず、Python 関数オブジェクトを保存するときに、参照カウントを次のようにインクリメントする必要があります。

Py_INCREF(pFunc)

そうしないと、Python はオブジェクト参照を保持していることを認識できず、ガベージ コレクションが行われる可能性があり、コールバックから使用しようとするとセグメンテーション エラーが発生する可能性があります。

次に、C/C++ コールバックが呼び出されたときにどのスレッドが実行されているかを考慮する必要があります。Python 以外で作成された別のスレッド (つまり、ソケットでデータを受信する C/C++ スレッド) からコールバックされる場合は、Python API 関数を呼び出す前に、Python の Global Interpreter Lock (GIL) を取得する必要があります。そうしないと、プログラムの動作が未定義になります。GIL を取得するには、次のようにします。

void callback() {
    PyGILState_STATE gstate;
    gstate = PyGILState_Ensure();

    // Get args, etc.

    // Call your Python function object
    PyObject * pInstance = PyObject_CallObject(pFunc, args);

    // Do any other needed Python API operations

    // Release the thread. No Python API allowed beyond this point.
    PyGILState_Release(gstate);
}

また、拡張モジュールの init 関数では、スレッドが適切に初期化されるように次のことを行う必要があります。

// Make sure the GIL has been created since we need to acquire it in our
// callback to safely call into the python application.
if (! PyEval_ThreadsInitialized()) {
    PyEval_InitThreads();
}

そうしないと、Python 以外のスレッドから GIL を取得しようとすると、クラッシュや奇妙な動作が発生する可能性があります。

詳細については、Python 以外で作成されたスレッドを参照してください。

于 2013-05-17T13:08:01.457 に答える
2

Python は実行元のディレクトリでモジュールを探す必要がありますが、Python がファイルを見つけられないことが問題であると思われる場合は、コンピュータ上の任意のディレクトリをプログラム内のモジュール検索パスに追加できます。 :

// Initialize the Python Interpreter
Py_Initialize();

// The following two lines to the trick:
// add path to your module to python's search paths
PyRun_SimpleString("import sys");
PyRun_SimpleString("sys.path.append(\"/path/to/python/module/here\")");

// Build the name object
pName = PyString_FromString("your_module");

// Load the module object
pModule = PyImport_Import(pName);

// pDict is a borrowed reference 
pDict = PyModule_GetDict(pModule);

// pFunc is also a borrowed reference 
pFunc = PyDict_GetItemString(pDict, "PlxMsgWrapper");

pArgs = ... 

if (PyCallable_Check(pFunc)) 
{
   PyObject_CallObject(pFunc, pArgs);
} else {
   PyErr_Print();
}
于 2013-05-17T12:10:54.173 に答える