编程语言
首页 > 编程语言> > boost :: python:使用回调

boost :: python:使用回调

作者:互联网

我有这样的头文件(fingisdk.h):

#ifndef FINGISDK_H_
#define FINGISDK_H_

#include "fingienum.h"

#ifdef __cplusplus
extern "C" {
#endif

typedef void (*fingi_event)(FINGI_EVENT_ID eventId, char* msg);

FINGI_EVENT_ID start_fingi_sdk(char* ini_file_location);
FINGI_EVENT_ID download_room_info(char** roominfo);

void register_fingi_sdk_status_event_listener(fingi_event pointer_to_listener_fnc);


#ifdef __cplusplus
}
#endif
#endif

然后我为此写了一个python包装器:

#include <fingisdk.h>
#include <fingienum.h>
#include <boost/python.hpp>


BOOST_PYTHON_MODULE(libpythonWrapper)
{
    using namespace boost::python;
    def("start_fingi_sdk", start_fingi_sdk);


}

和用于调用它的python文件是这样的:

import libpythonWrapper

print libpythonWrapper.start_fingi_sdk('file_location.ini')

到目前为止,这很好.但是,我不知道如何公开双精度双指针
功能:

FINGI_EVENT_ID download_room_info(char** roominfo);

和回调函数:

void register_fingi_sdk_status_event_listener(fingi_event pointer_to_listener_fnc);

谁能指出我一些文档或帮助我解决问题

谢谢

编辑1:

经过一番弄乱后,我想出了如何做指向指针函数的指针. Python不支持指针,因此必须包装download_room_info(char ** roominfo)以返回简单的字符串:

std::string download_room_info_python_wrapper() {
    char * value;
    FINGI_EVENT_ID result = download_room_info(&value);
    return std::string(value);
}

接着:

def("download_room_info", download_room_info_python_wrapper);

仍在寻找回调函数的解决方案

解决方法:

因此,您想将此API绑定到Python:

typedef void (*fingi_event)(FINGI_EVENT_ID eventId, char* msg);
void register_fingi_sdk_status_event_listener(fingi_event);

这是一个注册函数,它接受一个带有eventId和一个字符串的函数. OK,让我们定义一个我们自己的(当然,在外部“ C”中,因为API是):

void my_callback(FINGI_EVENT_ID eventId, char* msg)
{
  // do something
}

现在我们可以为register函数编写一个包装器:

void my_register()
{
  register_fingi_sdk_status_event_listener(my_callback);
}

并绑定它:

def("my_register", my_register);

那应该一切正常.但这没有用,因为我们实际上没有在回调中做任何事情.因此,这导致了一个子问题,即如何在回调中做任何事情?我的一个想法是,您应该建立自己的函数注册机制,该机制使您可以将Python回调函数注册到诸如全局PyObject之类的东西中,它将被设置为Python函数并使用Boost Python进行调用:

boost::python::object g_callback;
void my_callback(FINGI_EVENT_ID eventId, char* msg)
{
  if (g_callback)
    g_callback(eventId, msg);
}

现在只需要让用户分配全局回调即可:

void set_callback(boost::python::object func)
{
  g_callback = func;
  register_fingi_sdk_status_event_listener(my_callback);
}

def("set_callback", set_callback);

然后在Python中:

def callback(eventId, msg):
  print eventId, msg

set_callback(callback)

我认为应该是这样.如果您的回调API像许多回调API一样支持“ void * userData”参数,那么整个事情会容易得多.然后,我们将使用它来存储PyObject或其他有用的东西.但是API缺少此功能,因此我们在某个地方使用全局变量来记住要调用的函数.

标签:boost-python,python,c-4
来源: https://codeday.me/bug/20191122/2057273.html