如何将 Objective C 函数的地址传递给 C 函数?

How to pass address of Objective C function to a C function?

我正在使用 PJSIP 进行 SIP 呼叫 ios 应用程序。由于库是用 C 编写的,因此一些回调 C 方法需要 Objective C 函数的地址。我试过

ua_cfg.cb.on_incoming_call = &[self on_incoming_call:<#(pjsua_acc_id)#> _:<#(pjsua_call_id)#> _:<#(pjsip_rx_data *)#>]

但当然它不起作用,因为我不知道将什么作为参数传递给函数,它也没有给我函数的地址。

我怎样才能让它发挥作用?期待这个的 C 函数如下所示

void (*on_incoming_call)(pjsua_acc_id acc_id, pjsua_call_id call_id,
                 pjsip_rx_data *rdata);

ObjectiveC 支持纯 C,所以为什么不创建一个 C 函数 on_incoming_call 然后调用 ObjectiveC 代码。

我很确定直接使用 objective C 是可行的,但我不会去那里。问题始于 objectiveC 中有额外的函数参数,而这些参数对您来说是不可见的。例如参数self.

所以请尝试以下操作:

void on_incoming_call(pjsua_acc_id acc_id, pjsua_call_id call_id, pjsip_rx_data *rdata) {
    [MyObjectiveCObject onIncomingCall...];
}

现在您可以看到您可能会遇到一些问题,其中 on_incoming_call 处于静态上下文中,而您的对象可能是也可能不是。我通常为这种情况创建单例。最简单的可能看起来像:

static MyReceiver *currentReceiver = nil;

+ (MyReceiver *)current {
    if(currentReceiver == nil) {
        currentReceiver = [[MyReceiver alloc] init];
        ua_cfg.cb.on_incoming_call = &on_incoming_call;
    }
    return currentReceiver;
}

在这种情况下,您现在可以调用

void on_incoming_call(pjsua_acc_id acc_id, pjsua_call_id call_id, pjsip_rx_data *rdata) {
    [[MyReceiver current] onIncomingCall...];
}