5

我试图在运行时获取一个方法,然后使用它的数据结构来调用它的实现。澄清一下,这是出于学习目的,而不是出于任何实际原因。所以这是我的代码。

#import <Foundation/Foundation.h>
#import <stdio.h>

#import <objc/runtime.h>

@interface Test : NSObject 
-(void)method;
@end

@implementation Test

-(void)method {
    puts("this is a method");
}

@end 

int main(int argc, char *argv[]) {

    struct objc_method *t = (struct objc_method*) class_getInstanceMethod([Test class], @selector(method));
    Test *ztest = [Test new];

    (t->method_imp)(ztest, t->method_name);

    [ztest release];
    return 0;
}

的定义struct objc_method如下(在objc/runtime.h中定义)

typedef struct objc_method *Method;

....

struct objc_method {
    SEL method_name                                          OBJC2_UNAVAILABLE;
    char *method_types                                       OBJC2_UNAVAILABLE;
    IMP method_imp                                           OBJC2_UNAVAILABLE;
}                                                            OBJC2_UNAVAILABLE;

但是,当我尝试编译代码时,出现此错误。

error: dereferencing pointer to incomplete type

但是当我将它添加到我的代码中(以显式声明一个 objc_method)时,它会按预期工作。

struct objc_method {
    SEL method_name;
    char *method_types;
    IMP method_imp;
};

typedef struct objc_method* Method;

有人可以向我解释为什么我的代码在我明确声明这个结构时有效,而不是当我从 objc/runtime.h 导入它时?它与 OBJC2_UNAVAILABLE 有什么关系吗?我找不到它的定义,但它是在我的环境中定义的。

编辑:

我跑去gcc -E code.m -o out.m查看 OBJC2_UNAVAILABLE 被替换为什么,结果发现 OBJC2_UNAVAILABLE 在我的环境中被定义为 __attribute__((unavailable)) 。有人可以解释这意味着什么,Method如果这种结构“不可用”,为什么仍然有效?

4

2 回答 2

6

我刚刚仔细查看了 objc 运行时标头,发现了问题所在,以及如何解决它:)

因此,如果您查看包含结构透明定义的文件区域,您会看到它位于以下内容的主体中:

#if !__OBJC2__

...

#endif

由于这个 IS 已定义,这意味着我们所指的结构实际上是前向声明且不透明的(因此不完整)。

我们必须做的是使用提供的函数来访问这些成员:

IMP method_getImplementation(Method method);
SEL method_getName(Method method);
void method_getReturnType(Method method, char *dst, size_t dst_len);

等等。

您可以在Runtime Reference Guide中查看访问器方法的完整列表和许多其他好东西。

希望能帮助到你!

于 2012-08-02T05:43:25.877 回答
5

它的字段是之前定义的,但它在 ObjC-2 中是不透明的类型。请改用运行时,不要自己定义字段:

int main(int argc, char *argv[]) {

    Method t = class_getInstanceMethod([Test class], @selector(method));
    Test * ztest = [Test new];

    IMP imp = method_getImplementation(t);

    typedef void (*fn)(id,SEL);
    fn f = (fn)imp;
    f(ztest,@selector(method));

    [ztest release];
    return 0;
}
于 2012-08-02T05:51:13.400 回答