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

不要在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 关键字来自动生成这些代码(注:Xcode 现在会默认自动添加 @synthesize 关键字)。于是 Objective-C 的程序员幸福了。大部分时候根本就不用写 getter 和 setter。

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

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

苹果在它的开发者文档库中有一个专门讲 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
-(id)init {
     self = [super init];
     if (self) {
          _count = [[NSNumber alloc] initWithInteger:0];
     }
     return self;
}

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

1
2
3
4
5
6
7
- (id)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 来启动刚刚录制的宏来做替换了。相当方便。

Posted by 唐巧 Aug 10th, 2011 iOS

关注我的「iOS开发」微信公众号,每天获得精选的 iOS 开发文章和创业心得:

时间: 2024-08-02 05:52:48

不要在init和dealloc函数中使用accessor的相关文章

for-关于 += 在被调用函数中时的问题。

问题描述 关于 += 在被调用函数中时的问题. int add(int a){int b = 0: b += 1;}int main (void){int a =0;int i = 0;for(i = 0;i<= 5;i++){printf(""%dn""add(a));}return 0;}如果通过循环调用函数的话, b += 1;怎么进行赋值? 解决方案 使用指针或引用都可以达到要求,一般教科书上会使用指针. void add(int *a){ *a =

如果在SPARK函数中使用UDF或UDAF

Spark目前已经内置的函数参见: Spark 1.5 DataFrame API Highlights: Date/Time/String Handling, Time Intervals, and UDAFs 如果在SPARK函数中使用UDF或UDAF, 详见示例 package cn.com.systex import scala.reflect.runtime.universe import org.apache.spark.SparkConf import org.apache.spa

c语言数组与函数-如何在函数中给已经在主函数中定义好的数组赋值

问题描述 如何在函数中给已经在主函数中定义好的数组赋值 已经在main()中定义了一个长度为20的数组,想在定义的函数中给数组赋值,但一直报错,请问如何修改? void arrin(int *arr) { int i; arr[]={1,1,2,2,3,3,4,5,6,5,6,7,7,8,8,9,9,0,0}; for(i=0;i<20;i++) printf("%d",arr[i]); } main() { int testarr[20]; ............ } 解决方

个人 平均相差天数-excel函数中统计问题,统计函数

问题描述 excel函数中统计问题,统计函数 表1是多名电工的维修单登记表含接单日期及交单日期,如何在表2中统计表1中某一人的工作效率(平均相差天数)和平均服务满意度 解决方案 在表一中,把该人的信息每天的情况汇集成一张表,在利用统计方法中的函数

c++问题-C++的类中怎么在一个函数中引用上一层的函数

问题描述 C++的类中怎么在一个函数中引用上一层的函数 在同一个类中怎么引用先定义的函数来定义函数?比如在一个时期类的定义中,一个日期加1的函数中怎么调用先定义的一个求是否闰年的函数. 解决方案 C++定义隐式转换函数将类转换为内部的一个成员变量c++函数默认参数是一个好的设计吗? 解决方案二: 直接在函数内调用就可以了.不知道你说的上一层是什么意思.是基类的函数还是集合类所属的对象的函数,前者直接调用,或者用 基类类名::函数名后者用构造函数传对象指针 解决方案三: 同一个类中的函数都是通过t

qmap-QT Qmap 在一个函数中定义,怎么在另一个函数中遍历

问题描述 QT Qmap 在一个函数中定义,怎么在另一个函数中遍历 50C void address_pool::set_address_pool(QString get_IP){ QString ip; ip= get_IP; qDebug()<<""IP""<<ip; QStringList str=ip.split(""); QStringList strlist= str.at(0).split("&qu

c语言中怎么把一个大小不确定的二维数组当作参数传入函数中

问题描述 c语言中怎么把一个大小不确定的二维数组当作参数传入函数中 c语言中怎么把一个大小不确定的二维数组当作参数传入函数中,取大神,取大神,取大神 解决方案 用VC++新建一个程序,默认生成的main函数定义如下 int mian(int argc, char* args[]) 这就是一个例子. 解决方案二: 一个表示长度的参数,一个指向二维数组的指针 解决方案三: fun(args[][],int rows,int cols) 解决方案四: void Func(int array[][10]

javascript-在函数中写 this.xx=function 和 var xx =function() 的区别

问题描述 在函数中写 this.xx=function 和 var xx =function() 的区别 亲问各位一个问题 JS代码如下 function Test(){ this.add = function(){ alert(1); } this.modAdd = function(){ this.add(); add(); // 以上两种均无法调用到ADD方法 // Uncaught TypeError: this.add is not a function } } var test =

在PHP的图形函数中显示汉字

函数|汉字|图形|显示 如何在PHP图形函数中显示汉字,大家已经问了很久了,但一直没有结果.终于在参考另一篇文章后,我试验出来了! 方法: 首先,决定要显示哪些汉字,如"计数器"这三个字. 到Word2000(97不知可不可以)中去,输入这三个字,另存为,选择编码格式,然后再在右边的编码类型里选择unicode utf-8,记住不要选错了.存成一个编码格式的文本文件,比如说文件名叫test.txt. 用一个16进制的编辑器打开这个文件,这时你会看到,里面有一堆16进制字符,一个汉字占三