事件驱动的模型少不了Callback。
中间件需要通过回调的方式,驱动上层应用程序做相应动作,这时就需要在C Extending中回调Python中的代码,举例如下:
1. C Extending中增加注册接口:
int core_register_listener( void (* func)(int, void *), void * user_data )
{
func( 1, user_data );
}
static void event_arrived( int event, void * user_data )
{
PyObject * _args;
_args = Py_BuildValue( "(i)", event );
PyEval_CallObject(( PyObject * )user_data, _args );
Py_DECREF( _args );
}
static PyObject * register_listener( PyObject * self, PyObject * args )
{
PyObject * _handler;
PyArg_Parse( args, "(O)", &_handler );
Py_XINCREF( _handler );
return Py_BuildValue( "i", core_register_listener( event_arrived, _handler ));
}
2. Python中注册回调函数:
def events_arrived( event ):
_debug.output( 'Core Event: [' + event + ']' )
libhello.register_listener( events_arrived )
如此,C中的事件信息就回调到Python中,可以继续处理。
此示例只演示了Python单线程中的回调函数注册流程,如果C Extending是基于多线程设计,那么还额外的处理,在PyInit_libhello中,加入:
PyEval_InitThreads();
启用多线程支持。
之后,在任何Callback中,需要调用PyGILState_Ensure和PyGILState_Release用于对Python的GIL进行申请和释放,保证代码的安全执行。