11

我正在做一些从 Mono C# 到 Obj-C 的互操作并遇到了这个问题。C# 代码需要传递一个回调——它使用函数指针来完成。我可以从 Obj-C 端获取函数指针并调用它,一切正常。但是我现在需要将该函数指针作为回调提供给第三方 API,该 API 使用块作为回调。我希望第三方调用 C# 函数 - 所以在某种程度上我试图将函数指针转换为块以便第三方可以运行它,或者制作某种桥梁 - 创建我自己的运行块该函数指针并将其提供给第三方。我似乎无法找到一种方法来做到这一点 - 我将如何生成一个包含要运行哪个函数的信息的块,然后将其提供给第三方。也许我还有其他选择?

编辑:将函数放在全局变量中可能会起作用,但我希望能够拥有大量这些,因为第三方 API 是异步的,我不希望它调用错误的回调。

我试过的代码:

typedef void (*DummyAction)(char * result);
typedef void (^DummyBlock)(char * result);

@interface FunctionToBlock : NSObject
{
    DummyAction function;
    DummyBlock block;
}

- (id) initWithFunction: (DummyAction) func;
- (DummyBlock) block;
@end

@implementation FunctionToBlock : NSObject
- (id) initWithFunction: (DummyAction) func {
    if (self = [super init]) {
        function = func;
        block = ^(char * result) {
            function(result);
        };
    }
    return self;
}

- (DummyBlock) block {
    return block;
}
@end

然后我运行这个

void RegisterCallback( char * text, DummyAction callback)
{
    FunctionToBlock *funcToBlock = [[FunctionToBlock alloc] initWithFunction : callback];
    funcToBlock.block(text);
}

它因 BAD_ACCESS 而失败。也许我做错了什么,因为我对 Obj-C 还不是很精通。如果直接运行,我可以确认回调是正常的,并且正在调用块,但它在函数(结果)行上失败。

4

3 回答 3

7

关于什么

void (*myFunc)(int x); // ... your function pointer

void (^myBlock)(int) = ^(int x) {
    myFunc(x);
};

然后myBlock是一个块,它捕获函数指针的值并在块执行时调用函数。


补充:我的建议是,根据您的代码,使用@property(并假设您使用 ARC 编译):

FunctionToBlock.h:

typedef void (*DummyAction)(char * result);
typedef void (^DummyBlock)(char * result);

@interface FunctionToBlock : NSObject
{
    DummyAction function; // Not really needed.
}

- (id) initWithFunction: (DummyAction) func;
@property(copy, nonatomic) DummyBlock block;   // "copy" is important here!

@end

FunctionToBlock.m:

#import "FunctionToBlock.h"

@implementation FunctionToBlock : NSObject
@synthesize block = _block; // Can be ommitted if you use Xcode 4.4 or later.

- (id) initWithFunction: (DummyAction) func
{
    if (self = [super init]) {
        function = func; // Not really needed.
        self.block = ^(char * result) {
            func(result); // Use "func", not "self->function", to avoid retain cycle.
        };
    }
    return self;
}
于 2012-12-17T12:24:20.247 回答
6

为什么不只是一个简单的功能

typedef void (*DummyAction)(char * result);
typedef void (^DummyBlock)(char * result);

DummyBlock functionToBlock(DummyAction func) {
    return [[^(char * result) {
                 func(result);
             } copy] autorelease];
}
于 2012-12-18T00:27:32.027 回答
0

块在引擎盖下是指向本地数据结构的指针。一旦您离开声明它的范围,一个块就会变得无效。范围是 init 中的 if 语句;一旦你离开,这个块就无效了。

您在这里以一种糟糕的方式打破了编码约定。首先,实例变量应该以下划线开头,这样每个人都可以看到你在做什么。最好在不声明实例变量的情况下使用属性。并且每个块属性都应该声明为“副本”。如果你这样做,一切都很好。

于 2016-02-19T13:54:36.573 回答