Objective-c 中继承实例变量

发布于 2024-12-21 06:48:03 字数 455 浏览 1 评论 0原文

在 Objective-c 2.0 中,为什么子类需要使用 self 关键字引用父类中的实例变量?

考虑这个例子:

// a.h
@interface MyClass : NSObject
@property (nonatomic, retain) Object *myObject;
@end

// a.m
@implementation MyClass
@synthesize myObject;
@end


// b.h
@interface AnotherClass : MyClass
@end

// b.m
@implementation AnotherClass
- (void) someMethod {
    // error
    // Object *obj = myObject;

    // works
    // Object *obj = self.myObject;
}
@end

In Objective-c 2.0 why do subclasses need to reference instance variables in parent classes using the self keyword?

Consider this example:

// a.h
@interface MyClass : NSObject
@property (nonatomic, retain) Object *myObject;
@end

// a.m
@implementation MyClass
@synthesize myObject;
@end


// b.h
@interface AnotherClass : MyClass
@end

// b.m
@implementation AnotherClass
- (void) someMethod {
    // error
    // Object *obj = myObject;

    // works
    // Object *obj = self.myObject;
}
@end

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。

评论(2

要走干脆点 2024-12-28 06:48:03

您实际上并没有定义变量,只是定义了一个属性(它隐式定义了一个私有变量)。由于属性只是方法,因此您需要点语法。请注意,self.property[self property] 相同。

要解决此问题,请指定一个变量。我将举一个例子,其中变量的名称与属性的名称不同。大多数人为两者选择相同的名字,但我喜欢让它们不同,这样我就能立即明白哪个是指哪个。

// a.h
@interface MyClass : NSObject {
    // Instance variables are "protected" by default, except if you
    // use @private or @public.
    Object *myObjectVar;
}

@property (nonatomic, retain) Object *myObject;
@end

// a.m
@implementation MyClass
@synthesize myObject = myObjectVar;
@end


// b.h
@interface AnotherClass : MyClass
@end

// b.m
@implementation AnotherClass
- (void) someMethod {
    // works
    Object *obj = myObjectVar;

    // works
    obj = self.myObject;

    // the same as self.myObject
    obj = [self myObject];
}
@end

请注意分配时的差异:如果分配给变量,则对象不会自动保留。但如果您使用以下属性,它就会被保留:

myObjectVar = someObject; // not retained, old object not released!
self.myObject = someObject; // old object released, new object retained
[self setMyObject:someObject]; // same as the line above

编辑: 提到合成的实例变量默认情况下是私有的,正如 @Jason Coco 所指出的。 @NSGod 是对的,普通实例变量默认是受保护的,而不是公共的,修复了这个问题。

You haven't actually defined a variable, you only defined a property (which implicitly defines a variable that is private). And since property are just method, you need the dot syntax. Note that self.property is the same as [self property].

To fix this, specify a variable. I'll give you an example where the variable has a different name than the property. Most people chose the same name for both but I like to have them differ so I immediately see which one is meant.

// a.h
@interface MyClass : NSObject {
    // Instance variables are "protected" by default, except if you
    // use @private or @public.
    Object *myObjectVar;
}

@property (nonatomic, retain) Object *myObject;
@end

// a.m
@implementation MyClass
@synthesize myObject = myObjectVar;
@end


// b.h
@interface AnotherClass : MyClass
@end

// b.m
@implementation AnotherClass
- (void) someMethod {
    // works
    Object *obj = myObjectVar;

    // works
    obj = self.myObject;

    // the same as self.myObject
    obj = [self myObject];
}
@end

Note the difference when you assign: if you assign to your variable the object is not retained automatically. But it is retained if you use the property:

myObjectVar = someObject; // not retained, old object not released!
self.myObject = someObject; // old object released, new object retained
[self setMyObject:someObject]; // same as the line above

Edit: Mentioned that the synthesized instance variables are private by default, as noted by @Jason Coco. And @NSGod is right that normal instance variables are protected by default rather than public, fixed that.

一笑百媚生 2024-12-28 06:48:03

如果您实际上在超类中声明了实例变量,而不是依赖新运行时合成实例变量的能力(除了合成访问器方法之外),则它们不会。请参阅 Objective-C 编程语言:运行时差异,了解有关实例变量综合的更多信息。

例如,为了能够直接引用实例变量,您需要将以下内容更改

@interface MyClass : NSObject
@property (nonatomic, retain) Object *myObject;
@end

@interface MyClass : NSObject {
// there is an implied @protected directive here
    Object *myObject;
}

@property (nonatomic, retain) Object *myObject;

@end

:默认情况下,实例变量是@protected,这意味着该类和任何子类都可以访问直接实例变量。 @protected ivars 与 @public ivars 的不同之处在于您无法使用 -> 访问它们。 @private ivar 只能由声明它们的类访问。请参阅 Objective-C 编程语言:实例变量的范围了解更多信息。

They don't, provided you actually declare an instance variable in the superclass, rather than rely on the new runtime's ability to synthesize the instance variable (in addition to synthesizing the accessor methods). See The Objective-C Programming Language: Runtime Difference for more info on instance variable synthesis.

For example, to be able to refer to the instance variable directly, you'd need to change the following:

@interface MyClass : NSObject
@property (nonatomic, retain) Object *myObject;
@end

to:

@interface MyClass : NSObject {
// there is an implied @protected directive here
    Object *myObject;
}

@property (nonatomic, retain) Object *myObject;

@end

By default, instance variables are @protected, meaning the class and any subclasses can access the instance variables directly. @protected ivars differ from @public ivars in that you can't access them using ->. @private ivars can only be accessed by the class that declares them. See The Objective-C Programming Language: The Scope of Instance Variables for more info.

~没有更多了~
我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
原文