不要在init和dealloc函数中使用accessor

本文探讨了Objective-C2.0中的dotsyntax特性,介绍了正确实现成员变量的getter和setter方法的重要性,尤其是在内存管理方面。并强调了在init和dealloc方法中避免使用accessor的方法。

原文

AUG 10TH, 2011

Objective-C 2.0 增加了dot syntax,用于简单地调用成员变量的accessor。相当于java的getter和setter。 因为正常情况下,写一个accessor对于初学者来说,还是挺容易犯错的。比如有一个NSString * 的成员变量叫name。一个错误的写法是:

1
2
3
-(void) setName:(NSString *)newName {
     name = newName;
}

Java同学肯定想不通上面的代码有什么错吧?原因是Objective-C需要自己负责内存的释放。 所以需要在改变reference之前,将原对象release,对新的对象,也需要retain一下,代码就改成这样:

1
2
3
4
-(void) setName:(NSString *)newName {
     [name release];
     name = [newName retain];
}

初学者可能以为这样就对了,其实还是有错,如果newName和name的指向的是同一个对象,并且这个对象retain count只有1的话。 那么name release之后,这个对象就被回收掉了。所以应该改成:

1
2
3
4
5
6
-(void) setName:(NSString *)newName {
     if (name != newName) {
          [name release];
          name = [newName retain];
     }
}

这样才算是一个正确的set函数,Java同学肯定被吓到了,虽然知道这么写,但这比Java麻烦多了。于是,Objective-C允许程序员使用 @property + @synthesize 关键字来自动生成这些代码。于是Objective-C的程序员幸福了。大部分时候根本就不用写getter和setter。

但是需要小心,Objective-C的accessor不能在init和dealloc函数中使用!如果你在dealloc中这么写,就有问题:

1
2
3
4
-(void)dealloc {
     self.name = nil;
     [super dealloc]
}

苹果在它的developer文档库中有一个专门讲cocoa的内存管理的文章: http://developer.apple.com/library/mac/documentation/Cocoa/Conceptual/MemoryMgmt/MemoryMgmt.pdf

文章的第16页有一节题目是:Don’t Use Accessor Methods in Initializer Methods and dealloc

文章说:你唯一不应该用Accessor的地方是init函数和delloc函数。在init函数中,对于一个_count成员变量应该像下面这样赋值:

1
2
3
4
5
6
7
-(void) init {
     self = [super init];
     if (self) {
          _count = [[NSNumber alloc] initWithInteger:0];
     }
     return self;
}

对于一个带参数的init函数,你应该实现成下面这样:

1
2
3
4
5
6
7
- (void) initWithCount:(NSNumber *)startingCount {
     self = [super init];
     if (self) {
          _count = [startingCount copy];
     }
     return self;
}

对于在dealloc中,对应的写法应该是调release:

1
2
3
4
- (void)dealloc {
     [_count release];
     [super dealloc];
}

但是比较郁闷的是,文章最后没有说为什么不能!去stackoverflow上搜了一下,比较不靠谱的说法是这样少一次函数调用,更快。比较靠谱的说法是:在init和dealloc中,对象的存在与否还不确定,所以给对象发消息可能不会成功。

顺便说一下, 当发现这个文章的时候,我们的代码中已经有了很多这样错误用法。虽然程序没有出现严重的内存问题,但是为了保险,还是打算一行一行改掉,后来我想,这个能不能用vim搞定呢? 于是我进vim,用qa启动宏录制,然后输入

:%s/self./[/g  再输入:%s/= nil/release]/g 

再输入q, 这样就可以用@a来启动刚刚录制的宏来做替换了。相当方便。

在Objective - C中,`alloc` `dealloc` 是用于对象内存管理的重要方法。 ### `alloc` 方法 `alloc` 方法用于创建一个新的对象实例并分配内存。其核心原理与Objective - C的内存管理核心概念“引用计数”紧密相关。当调用 `alloc` 方法时,系统会为对象分配一块足够的内存空间,并且将该对象的引用计数初始化为1。这意味着对象在创建后就有了一次引用,因为它已经被分配到了内存中等待使用。 例如,以下代码展示了如何使用 `alloc` 方法创建一个对象: ```objective-c NSObject *myObject = [[NSObject alloc] init]; ``` 在这个例子中,`[[NSObject alloc] init]` 首先调用 `alloc` 方法为 `NSObject` 类的对象分配内存,然后调用 `init` 方法进行对象的初始化。创建后的 `myObject` 引用计数为1,表示它当前被使用了一次。 ### `dealloc` 方法 `dealloc` 方法则是用于释放对象所占用的内存。当一个对象的引用计数变为0时,系统会自动调用该对象的 `dealloc` 方法。在 `dealloc` 方法中,对象会释放它所拥有的所有资源,如释放它所引用的其他对象、关闭打开的文件描述符等。 以下是一个简单的 `dealloc` 方法示例: ```objective-c @interface MyClass : NSObject @end @implementation MyClass - (void)dealloc { // 释放对象所拥有的资源 [super dealloc]; } @end ``` 在这个例子中,当 `MyClass` 对象的引用计数变为0时,`dealloc` 方法会被调用。`[super dealloc]` 这行代码是为了确保父类的资源也能被正确释放。 ### 原理总结 `alloc` `dealloc` 是基于引用计数机制工作的。`alloc` 负责对象的创建引用计数的初始化,而 `dealloc` 则在引用计数为0时被调用,负责释放对象的资源内存。这种机制确保了对象在有引用时一直存在于内存中,而在没有引用时能及时释放内存,避免了内存泄漏。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值