是否可以在子类中重新声明私有ivars以使其可访问?

uch*_*aka 6 cocoa cocoa-touch subclass objective-c foundation

当继承在.h文件中列出私有ivars的Apple类时,是否可以在子类.m文件中的类扩展@interface中重新声明自己子类中的那些ivar,以使它们可以被子类实现访问?

小智 8

您必须记住,在子类中声明实例变量不会重新声明具有相同名称的现有超类实例变量,无论声称的重新声明是在公共接口还是类扩展中.即使您使用相同的名称,它也是一个不同的实例变量.

你可以自己试试.例如:

@interface Base : NSObject {
@private
    int _number;
}
@end

@implementation Base
- (id)init { self = [super init]; if (self) _number = 10; return self; }
- (void)logNumber { printf("base = %d\n", _number); }
@end

@interface Derived : Base
@end

@interface Derived () {
    int _number;
}
@end

@implementation Derived
- (id)init { self = [super init]; if (self) _number = 20; return self; }
- (void)logNumberDerived { printf("derived = %d\n", _number); }
@end

int main(void) {
    Derived *o = [Derived new];
    [o logNumber];
    [o logNumberDerived];
    return 0;
}
Run Code Online (Sandbox Code Playgroud)

输出:

base = 10
derived = 20
Run Code Online (Sandbox Code Playgroud)

因为_number在超类中与_number子类(扩展名)不同.如果检查二进制输出中的符号nm -a,您会注意到编译器生成两个不同的符号:

s _OBJC_IVAR_$_Base._number
s _OBJC_IVAR_$_Derived._number
Run Code Online (Sandbox Code Playgroud)