内存管理规则
本文总结了Objective-C中内存管理的规则。以下是基本规则:您只能释放或自动释放您所拥有的对象。如果您使用名字以“alloc”或“new”开头或名字中包含“copy”的方法(例如alloc,newObject或mutableCopy)创建了一个对象,则您会获得该对象的所有权;或者如果您向一个对象发送了一条retain消息,则您也会获得该对象的所有权。 您可以使用释放release或自动释放autorelease来释放一个对象的所有权。自动释放autorelease的意思是“将会发送释放release消息”(要了解究竟何时发送,请参考“自动释放池”)。下面的规则是基本规则的衍生,或者是用于处理特殊边界情况的规则:作为基本规则的推论,如果您需要将接收到的对象存储为某个实例变量的属性,您必须保留或复制该对象。(对于弱引用来说并不是这样,详见“对象的弱引用”,但这种情况一般是很罕见的。)通常,您应该将这部分工作交给存取方法(参考“存取方法”)来处理。被接收的对象通常要保证在接收它的方法中仍然有效,并且该方法也可以将对象安全地返回给它的调用者。例外的情况包括多线程应用程序和一些“分布式对象”环境,然而,此时您还必须注意自己是否修改了您用来接收其它对象的对象(见“共享对象的有效性”)。在必要的时候,将保留retain与释放release或自动释放autorelease组合使用,可以防止对象因为消息的正常的边界效应而失效。“对象的所有权和销毁”讨论了这些规则背后的推理过程。重要:Core Foundation对象也有类似的内存管理规则(参考Core Foundation内存管理编程指南)。但是,Cocoa和Core Foundation的命名规范是不同的。特别地,Core Foundation的“Core Foundation的内存管理编程指南”中的创建规则”并不适用于返回Objective-C对象的方法。例如,在下面的代码片段中,您无需负责释放myInstance的所有权:
MyClass *myInstance = [MyClass createInstance]; 参考“Cocoa中Core Foundation对象的内存管理”。 

解决方案 »

  1.   

    对象的所有权和销毁
    对一个程序来说,最好是尽可能少地使用内存。因此,Objective-C环境中定义了一些机制和策略,允许您对程序的内存进行管理。虽然您可以从底层实现的角度去考虑Objective-C程序的内存管理(详见“幕后:保留计数”),但是通常从对象所有权的角度来考虑这个问题会更容易。在 Objective-C 程序中,对象会被创建和销毁。为了确保您的应用程序不会使用不必要的内存,对象应该在不需要它们的时候被销毁。当然,在需要对象时保证它们不被销毁也很重要。为了满足这些需求,Cocoa定义了一种机制—对象所有权,通过该机制您可以指定您何时需要使用一个对象,又在何时完成对该对象的使用。为了充分理解对象所有权策略在Cocoa中是如何实现的,您还需要阅读“自动释放池”这部分的内容。对象所有权策略
    任何对象都可能拥有一个或多个所有者。只要一个对象至少还拥有一个所有者,它就会继续存在。如果一个对象没有所有者,则运行时系统会自动销毁它(参考“回收对象”)。为了确保您清楚自己何时拥有一个对象而何时不拥有对象,Cocoa设置了以下策略:任何您自己创建的对象都归您所有。您可以使用名字以“alloc”或“new”开头或名字中包含“copy”的方法(例如alloc,newObject,或mutableCopy)来“创建”一个对象。您可以使用retain来获得一个对象的所有权。请记住,一个对象的所有者可能不止一个。拥有一个对象的所有权就表示您需要保持该对象存在。(“存取方法”更详细地讨论了这部分内容。)当您不再使用您所拥有的对象时,您必须释放对这些对象的所有权。您可以通过向一个对象发送release消息或autorelease消息(在“自动释放”这一部分更详细地讨论了autorelease)来释放您对它的所有权。因此,用Cocoa的术语来说,释放对象的所有权通常被称为“释放”(releasing)对象。您不能释放非您所有的对象的所有权。这主要是前面的策略规则的一个隐含的推论,在这里将其明确地提出。这条策略对基于GUI的Cocoa应用程序和命令行Foundation工具都适用。请仔细思考下面的代码片段:{ 
        Thingamajig *myThingamajig = [[Thingamajig alloc] init]; 
        // ... 
        NSArray *sprockets = [myThingamajig sprockets]; 
        // ... 
        [myThingamajig release]; 
    } 这个例子完全遵守上述策略。您使用alloc方法创建了Thingamajig对象,因此,随后您在不需要该对象时对其发送了一条release消息。当您通过Thingamajig对象获得sprockets数组时,您并没有“创建”这个数组,所以您也没有对其发送release消息。幕后:保留计数
    所有权策略是在调用retain方法后通过引用计数—通常被称为“保留计数”—实现的。每个对象都有一个保留计数。当您创建一个对象时,该对象的保留计数为1。当您向一个对象发送retain消息时,该对象的保留计数加1。当您向一个对象发送release消息时,该对象的保留计数减1。当您向一个对象发送autorelease消息时,该对象的保留计数会在将来的某个阶段减1。如果一个对象的保留计数被减为0,该对象就会被回收(请参考“回收对象”)。重要:通常您不必显式地查询对象的保留计数是多少(参考retainCount)。其结果往往容易对人产生误导,因为您可能不知道您感兴趣的对象由何种框架对象保留。在调试内存管理的问题上,您只需要确保您的代码遵守所有权规则。
    自动释放
    NSObject对象定义的autorelease方法为后续的释放标记了接收者。通过向对象发送autorelease消息,您亦是声明:在您发送消息的作用域之外,您不想再保留该对象。这个作用域的范围是由当前的自动释放池定义的,可参考“自动释放池”。您可以这样实现上面提到的sprockets方法:– (NSArray *)sprockets { 
      
        NSArray *array = [[NSArray alloc] initWithObjects:mainSprocket, 
                                   auxiliarySprocket, nil]; 
        return [array autorelease]; 
    } 使用alloc创建数组;因此您将拥有该数组,并负责在使用后释放所有权。而释放所有权就是使用autorelease完成的。当另一个方法得到Sprocket对象的数组时,这个方法可以假设:当不在需要该数组时,它会被销毁,但仍可以在其作用域内的任何地方被安全地使用(请参考“共享对象的有效性”)。该方法甚至可以将数组返回给它的调用者,因为应用程序对象为您的代码定义好了调用堆栈的底部。autorelease方法可以使您很轻松地从一个方法返回一个对象,并且仍然遵循所有权策略。为了说明这一点,来看两个sprockets方法的错误实现:这种做法是错误的。根据所有权策略,这将会导致内存泄漏。– (NSArray *)sprockets { 
      
        NSArray *array = [[NSArray alloc] initWithObjects:mainSprocket, 
                                   auxiliarySprocket, nil]; 
        return array; 
    } 对象只能在sprockets方法的内部引用新的数组对象。在该方法返回后,对象将失去对新对象的引用,导致其无法释放所有权。这本身是没有问题的。但是,按照先前提出的命名约定,调用者并没有得到任何提示,不知道它已经获得了返回的对象。因此调用者将不会释放返回的对象的所有权,最终导致内存泄漏。这种做法也是错误的。虽然对象正确地释放了新数组的所有权,但是在release消息发送之后,新数组将不再具有所有者,所以它会被系统立即销毁。因此,该方法返回了一个无效(已释放)的对象:– (NSArray *)sprockets { 
      
        NSArray *array = [[NSArray alloc] initWithObjects:mainSprocket, 
                                   auxiliarySprocket, nil]; 
        [array release]; 
        return array; // array is invalid here 
    } 最后,您还可以像这样正确地实现sprockets方法:– (NSArray *)sprockets { 
      
        NSArray *array = [NSArray arrayWithObjects:mainSprocket, 
                                   auxiliarySprocket, nil]; 
        return array; 
    } 您并没有拥有arrayWithObjects:返回的数组,因此您不用负责释放所有权。不过,您可以通过sprockets方法安全地返回该数组。重要:要理解这一点,很容易令人联想到arrayWithObjects:方法本身正是使用autorelease实现的。虽然在这种情况下是正确的,但严格来讲它属于实现细节。正如您不必关心一个对象的实际保留计数一样,您同样不必关心返回给您的对象是否会自动释放。您唯一需要关心的是,您是否拥有这个对象。
    共享对象的有效性
    Cocoa的所有权策略规定,被接收的对象通常应该在整个调用方法的作用域内保持有效。此外,还可以返回从当前作用域接收到的对象,而不必担心它被释放。对象的getter方法返回一个缓存的实例变量或者一个计算值,这对您的应用程序来说无关紧要。重要的是,对象会在您需要它的这段期间保持有效。这一规则偶尔也有一些例外情况,主要可以总结为以下两类。当对象从一个基本的集合类中被删除的时候。heisenObject = [array objectAtIndex:n]; 
    [array removeObjectAtIndex:n]; 
    // heisenObject could now be invalid. 当对象从一个基本的集合类中被删除时,它会收到一条release(不是autorelease)消息。如果该集合是这个被删除对象的唯一所有者,则被删除的对象(例子中的heisenObject)将被立即回收。当一个“父对象”被回收的时候。id parent = <#create a parent object#>; 
    // ... 
    heisenObject = [parent child] ; 
    [parent release]; // Or, for example: self.parent = nil; 
    // heisenObject could now be invalid. 在某些情况下,您通过另外一个对象得到某个对象,然后直接或间接地释放父对象。如果释放父对象会使其被回收,而且父对象是子对象的唯一所有者,那么子对象(例子中的heisenObject)将同时被回收(假设它在父对象的dealloc方法中收到一条release而非autorelease消息)。为了防止这些情况发生,您要在接收heisenObject后保留该对象,并在使用完该对象后对其进行释放,例如:heisenObject = [[array objectAtIndex:n] retain]; 
    [array removeObjectAtIndex:n]; 
    // use heisenObject. 
    [heisenObject release]; 存取方法
    如果您的类中有一个实例变量本身是一个对象,那么您必须保证任何为该实例变量赋值的对象在您使用它的过程中不会被释放。因此,您必须在对象赋值时要求获取它的所有权。您还必须保证在将来会释放任何您当前持有的值的所有权。例如,如果您的对象允许设置它的main Sprocket,您可以这样实现setMainSprocket:方法:– (void)setMainSprocket:(Sprocket *)newSprocket { 
        [mainSprocket autorelease]; 
        mainSprocket = [newSprocket retain]; /* Claim the new Sprocket. */ 
        return; 
    } 现在,setMainSprocket:可能在被调用时带一个Sprocket对象的参数,而调用者想要保留该Sprocket对象,这意味着您的对象将与其他对象共享Sprocket。如果有其他对象修改了Sprocket,您的对象的main Sprocket也会发生变化。但如果您的Thingamajig需要有属于它自己的Sprocket,您可能会认为该方法应该复制一份私有的副本(您应该记得复制也会得到所有权):– (void)setMainSprocket:(Sprocket *)newSprocket { 
        [mainSprocket autorelease]; 
        mainSprocket = [newSprocket copy]; /* Make a private copy. */ 
        return; 
    } 以上几种实现方法都会自动释放原来的main sprocket。如果newSprocket和mainSprocket是同一个的对象,并且Thingamajig对象是它的唯一所有者的话,这样做可以避免一个可能出现的问题:在这种情况下,当sprocket被释放时,它会被立即回收,这样一旦它被保留或复制,就会导致错误。下面的实现也解决了这个问题:– (void)setMainSprocket:(Sprocket *)newSprocket { 
        if (mainSprocket != newSprocket) { 
            [mainSprocket release]; 
            mainSprocket = [newSprocket retain]; /* Or copy, if appropriate. */ 
        } 
    } 在所有这些情况中,看起来好像最终为您的对象设置的mainSprocket泄漏了,因为您不用释放对它的所有权。这些由dealloc方法负责,在“回收对象”部分进行了介绍。“存取方法”部分更详细地描述了存取方法及其实现。 回收对象
    当一个对象的保留计数减少至0时,它的内存将被收回—在Cocoa术语中,这被称为“释放”(freed)或“回收”(deallocated)。当一个对象被回收时,它的dealloc方法被自动调用。dealloc方法的作用是释放对象占用的内存,释放其持有的所有资源,包括所有实例变量对象的所有权。如果在您的类中有实例变量对象,您必须实现一个dealloc方法来释放它们,然后调用超类的dealloc实现。例如,如果Thingamajig类含有mainSprocket和auxiliarySprocket实例变量,您应该这样实现该类的dealloc方法:- (void)dealloc { 
        [mainSprocket release]; 
        [auxiliarySprocket release]; 
        [super dealloc]; 
    } 重要:决不要直接调用另一个对象的dealloc方法。您不应该让系统资源的管理依赖于对象的生命周期;参考“资源管理”。当应用程序终止时,对象有可能没有收到dealloc消息。由于进程的内存在退出时被自动清空,因此与调用一切内存管理方法相比,简单地让操作系统清理资源效率更高。
    通过引用返回的对象
    Cocoa的一些方法可以指定一个通过引用(即ClassName ** 或 id *)返回的对象。下面有几个使用NSError对象的例子,该对象包含有错误出现时的信息,例如:initWithContentsOfURL:options:error: (NSData)initWithContentsOfFile:encoding:error: (NSString)executeFetchRequest:error: (NSManagedObjectContext)在这些情况下,前面描述的规则同样适用。当您调用这些方法中的任何一种时,由于您没有创建NSError对象,因此您不会拥有该对象—同样也无需释放它。NSString *fileName = <#Get a file name#>; 
    NSError *error = nil; 
    NSString *string = [[NSString alloc] initWithContentsOfFile:fileName 
                            encoding:NSUTF8StringEncoding error:&error]; 
    if (string == nil) { 
        // deal with error ... 

    // ... 
    [string release]; 如果因为任何原因,返回的对象的所有权不能遵守基本规则,这将在方法的文档中明确地阐明(例如,参考dataFromPropertyList:format:errorDescription:)。保留循环 
    在某些情况下,两个对象之间可能会出现循环引用的情况,也就是说,每一个对象都包含一个实例变量引用对方对象。例如,考虑一个文本程序,程序中对象间的关系如图1所示。“文档(Document)”对象为文档中的每个页面创建一个“页(Page)”对象。每个Page对象具有一个实例变量,用来跟踪该页所在的文档。如果Document对象保留了Page对象, 同时Page对象也保留Document对象,则这两个对象都永远不会被释放。只有Page对象被释放,Document的引用计数才能变为0,而只有Document对象被回收,Page对象才能被释放。图 1  保留循环示意图
    针对保留循环问题的解决方案是“父”对象应保留其“子”对象,但子对象不应该保留其父对象。因此,在图1中,document对象要保留page对象,但page对象不保留document对象。子对象对其父对象的引用是一个弱引用的例子,这部分内容在“对象的弱引用”有更充分的描述。对象的弱引用
    保留一个对象创建了一个对该对象的“强”引用。一个对象只有在它的所有强引用都被释放后才能被回收。因此,一个对象的生命周期取决于其强引用的所有者。在某些情况下,这种行为可能并不理想。您可能想要引用一个对象而不妨碍对象本身的回收。对于这种情况,您可以获取一个“弱”引用。弱引用是通过存储一个指向对象的指针创建的,而不是保留对象。弱引用在可能会出现循环引用的情况下是必不可少的。例如,如果对象A和对象B互相通信,两者都需要引用对方。如果每个对象都保留对方对象,则这两个对象只有在它们之间的连接中断后才能被回收,但是它们之间的连接又只能在有对象被回收后才能中断。为了打破这种循环,其中一个对象需要扮演从属角色,得到另一个对象的一个弱引用。举个具体的例子,在视图层次中,父视图拥有其子视图,也因此能够保留子视图,但父视图并不归子视图所有;然而子视图仍需要知道谁是它的父视图,因此它保持一个对其父视图的弱引用。Cocoa中弱引用的其他适用情况包括:表格数据源,大纲视图项,通知观察者以及其余项目标和委托,但不仅限于上述情况。 重要:在Cocoa中,引用表格数据源,大纲视图项,通知观察者和委托都被看作是弱引用(例如,NSTableView对象不保留其数据源,NSApplication对象不保留它的委托)。本文档仅仅介绍了这一公约的例外情况。
    在向您弱引用的对象发送消息时,您需要小心谨慎。如果您在一个对象被回收之后向它发送消息,您的应用程序将会崩溃。您必须为对象何时有效制定有明确界定的条件。在大多数情况下,被弱引用的对象知道其他对象对它的弱引用,这和循环引用的情况是一样的,并且它还能够在自己被回收时通知其他对象。例如,当您向通知中心注册一个对象的时候,通知中心会存储一个对该对象的弱引用,并且在适当的消息发布时,还会向该对象发送消息。当对象被回收时,您需要向通知中心解注册该对象,以防通知中心向这个已经不存在的对象继续发送消息。同样,当一个委托对象被回收时,您需要通过向其他对象发送一条带nil参数的setDelegate:消息来删除委托链接。这些消息通常由对象的dealloc方法发出。资源管理
    通常,不应该由您来管理一些稀缺资源,比如文件描述符,网络连接和dealloc方法中的缓冲区/高速缓存。特别地,您设计的类不应该让您错误地认为dealloc会在您觉得该调用的时候被调用。dealloc的调用可能会因为bug或应用程序销毁而被延误或回避。相反,如果您有一个类,由它的实例管理稀缺资源,则您在设计应用程序时应该让自己知道何时不再需要这些资源,并且可以在那个时刻通知实例“清理”这些资源。通常,接下来您应该释放该实例,然后调用dealloc,但如果您不这样做也不会有问题。如果您试图让dealloc肩负起资源管理的责任,会出现的一些问题:对象图销毁的顺序依赖性。对象图销毁机制内在是无序的。虽然通常您可能期望甚至得到一个特定的顺序,但是这样做的同时您也引入了脆弱性。如果一个对象意外落入自动释放池,则销毁顺序会改变,这可能会导致意想不到的后果。稀缺资源的未回收。内存泄露当然是应该被修复的bug,但它们一般来说不会是直接致命的错误。然而,如果稀缺资源在您希望它们被释放时没有被释放,这会导致非常非常严重的问题。例如,如果您的应用程序耗尽了文件描述符,那么用户可能无法保存数据。清除在错误的线程上被执行的逻辑。如果一个对象在意外的时刻落入自动释放池,那么无论它进入哪个线程池,它都将被回收。这对于那些本应该只由一个线程访问的资源来说很容易产生致命的错误。
      

  2.   

    实用内存管理
    本文为您提供了一种透视内存管理的实用性视角。这部分内容涵盖了“对象所有权和销毁”中介绍的基本概念,不过采用了更加面向代码实现的视角。遵从以下几条简单的规则可以使内存管理变得更加容易,而不遵守这些规则将几乎肯定会在某些时候导致内存泄漏,或者由于消息被发送给已释放的对象而导致运行时异常。基础知识
    为了让应用程序的内存消耗尽可能低,您应该清除掉不使用的对象,但是您需要确保您清除的不是正在被使用的对象。因此,您需要一种机制,可以让您标记那些仍然有用的对象。所以,从许多方面来讲,站在“对象的所有权”的角度看内存管理是最好理解的。一个对象可以有一个或一个以上的所有者。采用类比的方式,您可以联想一个分时租用公寓。当一个对象没有所有者的时候,它会被销毁。继续用类比的方法,您可以联想一个分时合租的寓所,但当地居民并不喜欢它。如果没有所有者,这处合租寓所将被拆除。为了确保您感兴趣的对象不被销毁,您必须成为它的一个所有者。您可以建造一所新公寓,或入住一所现有的公寓。为了让您不再感兴趣的对象能够被销毁,您应该释放它的所有权。您可以出售您的分时租用公寓。为了支持这个模型,Cocoa提供了一种被称为“引用计数”或“保留计数”的机制。每一个对象都有一个保留计数。当对象被创建的时候,其保留计数为1。当保留计数减少至0时,对象会被回收(销毁)。您可以使用各种方法来操作保留计数(即获取或释放所有权):alloc
    为对象分配内存并返回该对象,其保留计数为1。您拥有以单词alloc或new开头的任意方法创建的对象。copy
    为对象创建一份副本并返回该对象,其保留计数为1。如果您复制一个对象,您就拥有了这个对象的副本。这对于任何名字中包含单词copy的方法都是适用的,这里的“copy”是指被返回的对象。retain
    使一个对象的保留计数增加1。获得一个对象的所有权。release
    使一个对象的保留计数减少1。释放一个对象的所有权。autorelease
    使一个对象的引用计数在未来的某个阶段减少1。在未来的某个阶段释放一个对象的所有权。内存管理的实用规则如下(也可以参考“内存管理规则”):您只拥有那些您使用名字以“alloc”或“new”开头或者名字中包含“copy”的方法(例如alloc,newObject或mutableCopy)创建的对象,或者是那些收到了您发送的retain消息的对象。许多类提供了形如+className...的方法,您可以使用它们获得该类的一个新的实例。这些方法通常被称为“简便构造函数”,它们创建一个新的类的实例,对其进行初始化并将其返回供您使用。您并不拥有从简便构造函数或其它存取方法返回的对象。一旦您使用完一个您拥有的对象,您应该使用release或autorelease释放这个对象的所有权。通常,您应该使用release,而不是autorelease。只有在不适合立即回收对象的情况下,您才应该使用autorelease,比如您要从某个方法返回对象。(注意:这并不是说release必然会引起对象的回收—只有当保留计数减少至0时才会发生回收—但它也有可能会发生,而有时您需要防止出现这种情况,请参考“从方法返回对象”中的例子。)实现dealloc方法来释放您拥有的实例变量。您不应该直接调用dealloc(除非是您在自定义的dealloc方法中调用超类的实现)。几个简单的例子
    下面几个简单的例子对比说明了使用alloc,简便构造函数和存取方法创建一个新对象。第一个例子使用alloc创建了一个新的字符串对象。因此,它必须被释放。- (void)printHello { 
        NSString *string; 
        string = [[NSString alloc] initWithString:@"Hello"]; 
        NSLog(string); 
        [string release]; 
    } 第二个例子使用简便构造函数创建了一个新的字符串对象。此外没有额外的工作要做。- (void)printHello { 
        NSString *string; 
        string = [NSString stringWithFormat:@"Hello"]; 
        NSLog(string); 
    } 第三个例子使用存取方法获取一个字符串对象。与简便构造函数一样,没有额外的工作要做。- (void)printWindowTitle { 
        NSString *string; 
        string = [myWindow title]; 
        NSLog(string); 
    } 使用存取方法
    虽然使用存取方法有时看似繁琐,有故意卖弄之嫌,但如果您坚持使用存取方法,则内存管理方面出现问题的可能性将大大减小。如果您在代码中对实例变量全部使用retain和release,那么几乎可以肯定您在做错误的事情。考虑一个“计数器”对象(Counter),您要设置它的计数。@interface Counter : NSObject { 
        NSNumber *count; 
    } 为了获取和设置计数的值,您需要定义了两个存取方法。(下面的例子给出了存取方法的简单实现。在“存取方法”中有对它们更加详细的介绍。)在get方法中,您只是回传了一个变量,所以没有必要进行retain或release:- (NSNumber *)count { 
        return count; 
    } 在set方法中,如果其他人都按照同样的规则进行操作,则您需要假设新的计数可能会在任何时刻被销毁,因此您需要获得对象的所有权—向它发送一条retain消息—来确保它不会被销毁。在这里您还需要通过向旧的计数对象发送一条release消息来释放它的所有权。(Objective-C中允许向nil发送消息,因此在计数尚未被设置时仍然可以这么做。)您必须在[newCount retain]之后发送这个消息,以防这两者是同一个对象—您肯定不希望由于疏忽造成对象意外被回收。- (void)setCount:(NSNumber *)newCount { 
        [newCount retain]; 
        [count release]; 
        // make the new assignment 
        count = newCount; 
    } 只有在两处地方您不该使用存取方法来设置实例变量—init方法和dealloc。为了用一个表示零的数字对象初始化一个计数对象,您可以按照下面的方式实现一个init方法:- init { 
        self = [super init]; 
        if (self) { 
            count = [[NSNumber alloc] initWithInteger:0]; 
        } 
        return self; 
    } 为了用一个非零的计数初始化计数器,您可以这样实现一个initWithCount:方法:- initWithCount:(NSNumber *)startingCount { 
        self = [super init]; 
        if (self) { 
            count = [startingCount copy]; 
        } 
        return self; 
    } 由于“计数器”类(Counter)有一个对象实例变量,您还必须实现一个dealloc方法。该方法应该可以通过向实例变量发送release消息来释放任何实例变量的所有权,并且最终调用超类的dealloc实现:- (void)dealloc { 
        [count release]; 
        [super dealloc]; 
    } 实现重置方法
    假设您想实现一个方法来重置计数器。那么您有两种选择,第一种是使用简便构造函数创建一个新的NSNumber对象—也因此没有必要发送任何retain或release消息。请注意,两种方法都使用了类的set存取方法。- (void)reset { 
        NSNumber *zero = [NSNumber numberWithInteger:0]; 
        [self setCount:zero]; 
    } 第二种是使用alloc创建NSNumber实例,因此您要相应地使用release。- (void)reset { 
        NSNumber *zero = [[NSNumber alloc] initWithInteger:0]; 
        [self setCount:zero]; 
        [zero release]; 
    } 常见错误
    下面几小节举例说明常见的错误。没有使用存取方法
    下面的例子在一些简单的情况下几乎肯定可以正常工作,但这个例子避免使用存取方法,这样做几乎肯定会在某个阶段(当您忘记保留或释放,或者当您的实例变量的内存管理语义发生变化的时候)导致错误。- (void)reset { 
        NSNumber *zero = [[NSNumber alloc] initWithInteger:0]; 
        [count release]; 
        count = zero; 
    } 另外请注意,如果您正在使用键值观察(参考键值观察编程指南),那么用这种方式改变变量是不兼容KVO的。实例泄露
    - (void)reset { 
        NSNumber *zero = [[NSNumber alloc] initWithInteger:0]; 
        [self setCount:zero]; 
    } 新数字的保留计数是1(来自alloc ),而且在该方法释放的作用域内没有与之对应的release。新数字是不可能被释放的,这将导致内存泄漏。向非您所有的实例发送 release
    - (void)reset { 
        NSNumber *zero = [NSNumber numberWithInteger:0]; 
        [self setCount:zero]; 
        [zero release]; 
    } 如果没有调用retain ,则在当前的自动释放池被释放后,下一次您访问count会失败。简便构造方法返回一个会自动释放的对象,所以您不必再发送release。这样做意味着,当因autorelease而产生的release被发送后,保留计数会被减为0,且对象将被释放。当您下次想要访问计数时,您将向一个已经被释放的对象发送消息(这时通常您会得到一个SIGBUS 10错误)。会造成混乱的情况
    使用集合
    当您把一个对象添加到一个集合,比如数组,字典或集合,集合拥有对象的所有权。当对象从集合中删除或集合本身被释放时,集合会释放所有权。因此,举例来说,如果您想创建一个数字数组,您可以选择以下方法中的一种:NSMutableArray *array; 
    NSUInteger i; 
    // ... 
    for (i = 0; i < 10; i++) { 
        NSNumber *convenienceNumber = [NSNumber numberWithInteger:i]; 
        [array addObject:convenienceNumber]; 
    } 在这段代码中,您没有调用alloc,因此也没有必要调用release。没有必要保留新的数字对象(convenienceNumber),因为数组会为您代劳。NSMutableArray *array; 
    NSUInteger i; 
    // ... 
    for (i = 0; i < 10; i++) { 
        NSNumber *allocedNumber = [[NSNumber alloc] initWithInteger: i]; 
        [array addObject:allocedNumber]; 
        [allocedNumber release]; 
    } 在这段代码中,您需要在for循环的作用域内向allocedNumber发送release消息,以抵消之前的alloc。由于数组在用addObject:方法添加数字时对其进行了保留,因此只要它还在数组中就不会被释放。要理解这一点,您要把自己放在实现这种集合类的作者的位置。您要确保交给您管理的对象不能在您的眼皮底下消失,所以您要在这些对象被加入集合中时向它们发送retain消息。如果它们被删除,您还必须相应地发送release消息,并且在您自己的dealloc方法中,您还应该向其余的对象发送release消息。从方法返回的对象
    当您从一个方法中返回一个局部变量时,您不仅要保证自己遵守了内存管理规则,而且要保证接收方在对象被释放之前一直有机会使用该对象。当您返回一个新创建的(您拥有的)对象时,您应该是用autorelease而不是release来释放所有权。请考虑一个很简单的fullName方法,用它来连接firstName和lastName。一种可行的正确的实现方法(仅仅从内存管理的角度而言—当然从功能性的角度考虑,它仍有很多不足之处)可能如下面的代码所示:- (NSString *)fullName { 
        NSString *string = [NSString stringWithFormat:@"%@ %@", firstName, lastName]; 
        return string; 
    } 按照最基本的规则,您并不拥有stringWithFormat返回的字符串,所以它可以安全地从该方法中返回。下面这种实现方法也是正确的:- (NSString *)fullName { 
        NSString *string = [[[NSString alloc] initWithFormat:@"%@ %@", firstName, lastName] autorelease]; 
        return string; 
    } 您拥有alloc返回的字符串,但您随后向它发送了一条autorelease消息,因此在您失去它的引用之前,您已经放弃了所有权,并且这样做也是满足内存管理规则的。这种实现方法的精髓在于使用了autorelease而不是release,要意识到这一点。相比之下,下面的代码是错误的 :- (NSString *)fullName { 
        NSString *string = [[[NSString alloc] initWithFormat:@"%@ %@", firstName, lastName] release]; 
        return string; 
    } 纯粹从内存管理的角度来讲,它看起来是正确的:您拥有alloc返回的字符串,并向它发送一条release的信息来释放所有权。然而从实用角度来看,该字符串很有可能在那一步就被回收了(它可能没有任何其他的所有者),因此该方法的调用者会接收到一个无效的对象。这说明了为什么autorelease非常实用—它能让您推迟释放,您可以在未来的某一时刻过后再释放对象。为了追求完整性,下面的代码也是错误的 :- (NSString *)fullName { 
        NSString *string = [[NSString alloc] initWithFormat:@"%@ %@", firstName, lastName]; 
        return string; 
    } 您拥有alloc返回的字符串,但是在您有机会释放所有权之前,您就失去了对该对象的引用。根据内存管理规则,这将导致内存泄漏,因为调用者没有得到任何迹象表明他们拥有返回的对象。