3

我刚刚开始学习 Objective-C,我正在阅读Stephen G. KochanProgramming in Objective-C 3rd Edition

有一段解释多态机制:

在运行时,Objective-C 运行时系统将检查存储在dataValue1(一个 id 对象)中的对象的实际类,并从正确的类中选择适当的方法来执行。但是,在更一般的情况下,编译器可能生成不正确的代码以将参数传递给方法或处理其返回值。例如,如果一个方法将对象作为其参数,而另一个方法采用浮点值,则会发生这种情况。或者,例如,如果一种方法返回一个对象,而另一种方法返回一个整数。如果两个方法之间的不一致只是对象类型不同(例如,Fraction 的 add: 方法将 Fraction 对象作为其参数并返回一个,而 Complex 的 add: 方法采用并返回一个 Complex 对象),编译器将仍然生成正确的代码,因为内存地址(即指针)无论如何都是作为对对象的引用传递的。

我不太明白该段的第一部分说如果我在具有相同名称和不同类型参数的不同类中声明 2 个方法,编译器可能会生成不正确的代码。而该段的最后一部分说可以有两个具有相同名称和不同参数和返回类型的方法......哦不......

我有以下代码,它们编译并运行良好:

@implementation A
- (int) add:(int)a {
    return 1 + a;
}
@end
@implementation B
- (int) add: (B*) b {
    return 100;
}
@end
id a = [[A alloc] init];
id b = [[B alloc] init];
NSLog(@"A: %i, B %i", [a add:100], [b add:b]);

编辑:正如我引用的文本,上面的代码应该会导致错误,但它只会产生一些警告消息,发现多个名为“add:”的方法指向整数转换的不兼容指针将“id”发送到“int”类型的参数

我有 Java 和 C++ 背景,我知道 Objective-C 中的多态性与那些语言的多态性略有不同,但我仍然对不确定性感到困惑(粗体文本)。

我想我一定是误解了一些东西,请您为我和需要它的人更详细地解释一下 Objective-C 中的动态绑定吗?

谢谢!

4

2 回答 2

6

您没有注意到任何异常,因为这两个方法在例如 x86_64 ABI 下具有相同的调用语义。指针可以被认为是整数,并且在 x86_64 ABI 下,它们以相同的方式传递给目标方法。

但是,如果您有其他课程,例如:

@implementation C
- (int)add:(float)number {
    return (int)number + 100;
}
@end

解析时接收浮点参数(如 Kochan 所述),然后是编译器:

id a = [[A alloc] init];
id b = [[B alloc] init];
id c = [[C alloc] init];
NSLog(@"A: %i, B %i, C %i", [a add:100], [b add:b], [c add:100]);

不知道[c add:100]它应该放在100x86_64 ABI 指定的浮点寄存器中。因此,-[C add:]期望浮点参数位于浮点寄存器中的 ,读取与100参数不对应的值。

要使其工作,您必须使用静态类型声明变量:

C *c = [[C alloc] init];

或在发送消息时将其转换为正确的类型:

[(C *)c add:100];

归根结底,发送 Objective-C 消息是一个函数调用。不同的 ABI 在调用具有可变参数、浮点与整数参数或返回值或结构而不是标量算术类型的函数时可能具有不同的语义。如果编译器看到根据目标 ABI 以不同方式处理的不同方法签名,并且如果没有足够的类型信息可用,它可能最终选择了错误的方法签名。

于 2011-11-06T04:54:50.427 回答
3

之所以有区别,是因为在后一种情况下,区别仅在于论点的类别。Complex*andFraction*都是指针,所以即使两个同名方法之间存在混淆,也没有问题。

另一方面,您在示例中遇到的情况很危险,因为一个参数是指针,另一个是int. 但是,确保安全很容易:

NSLog(@"A: %i, B %i", [(A*)a add:100], [(B*)b add:b]);
于 2011-11-06T04:25:56.233 回答