Understanding reference counting with Cocoa and Objective-C

后端 未结 14 1541
攒了一身酷
攒了一身酷 2020-11-22 16:46

I\'m just beginning to have a look at Objective-C and Cocoa with a view to playing with the iPhone SDK. I\'m reasonably comfortable with C\'s malloc and f

相关标签:
14条回答
  • 2020-11-22 17:21

    If you're writing code for the desktop and you can target Mac OS X 10.5, you should at least look into using Objective-C garbage collection. It really will simplify most of your development — that's why Apple put all the effort into creating it in the first place, and making it perform well.

    As for the memory management rules when not using GC:

    • If you create a new object using +alloc/+allocWithZone:, +new, -copy or -mutableCopy or if you -retain an object, you are taking ownership of it and must ensure it is sent -release.
    • If you receive an object in any other way, you are not the owner of it and should not ensure it is sent -release.
    • If you want to make sure an object is sent -release you can either send that yourself, or you can send the object -autorelease and the current autorelease pool will send it -release (once per received -autorelease) when the pool is drained.

    Typically -autorelease is used as a way of ensuring that objects live for the length of the current event, but are cleaned up afterwards, as there is an autorelease pool that surrounds Cocoa's event processing. In Cocoa, it is far more common to return objects to a caller that are autoreleased than it is to return objets that the caller itself needs to release.

    0 讨论(0)
  • 2020-11-22 17:22

    If you understand the process of retain/release then there are two golden rules that are "duh" obvious to established Cocoa programmers, but unfortunately are rarely spelled out this clearly for newcomers.

    1. If a function which returns an object has alloc, create or copy in its name then the object is yours. You must call [object release] when you are finished with it. Or CFRelease(object), if it's a Core-Foundation object.

    2. If it does NOT have one of these words in its name then the object belongs to someone else. You must call [object retain] if you wish to keep the object after the end of your function.

    You would be well served to also follow this convention in functions you create yourself.

    (Nitpickers: Yes, there are unfortunately a few API calls that are exceptions to these rules but they are rare).

    0 讨论(0)
  • 2020-11-22 17:24

    Objective-C uses Reference Counting, which means each Object has a reference count. When an object is created, it has a reference count of "1". Simply speaking, when an object is referred to (ie, stored somewhere), it gets "retained" which means its reference count is increased by one. When an object is no longer needed, it is "released" which means its reference count is decreased by one.

    When an object's reference count is 0, the object is freed. This is basic reference counting.

    For some languages, references are automatically increased and decreased, but objective-c is not one of those languages. Thus the programmer is responsible for retaining and releasing.

    A typical way to write a method is:

    id myVar = [someObject someMessage];
    .... do something ....;
    [myVar release];
    return someValue;
    

    The problem of needing to remember to release any acquired resources inside of code is both tedious and error-prone. Objective-C introduces another concept aimed at making this much easier: Autorelease Pools. Autorelease pools are special objects that are installed on each thread. They are a fairly simple class, if you look up NSAutoreleasePool.

    When an object gets an "autorelease" message sent to it, the object will look for any autorelease pools sitting on the stack for this current thread. It will add the object to the list as an object to send a "release" message to at some point in the future, which is generally when the pool itself is released.

    Taking the code above, you can rewrite it to be shorter and easier to read by saying:

    id myVar = [[someObject someMessage] autorelease];
    ... do something ...;
    return someValue;
    

    Because the object is autoreleased, we no longer need to explicitly call "release" on it. This is because we know some autorelease pool will do it for us later.

    Hopefully this helps. The Wikipedia article is pretty good about reference counting. More information about autorelease pools can be found here. Also note that if you are building for Mac OS X 10.5 and later, you can tell Xcode to build with garbage collection enabled, allowing you to completely ignore retain/release/autorelease.

    0 讨论(0)
  • 2020-11-22 17:26

    I'll not add to the specific of retain/release other than you might want to think about dropping $50 and getting the Hillegass book, but I would strongly suggest getting into using the Instruments tools very early in the development of your application (even your first one!). To do so, Run->Start with performance tools. I'd start with Leaks which is just one of many of the instruments available but will help to show you when you've forgot to release. It's quit daunting how much information you'll be presented with. But check out this tutorial to get up and going fast:
    COCOA TUTORIAL: FIXING MEMORY LEAKS WITH INSTRUMENTS

    Actually trying to force leaks might be a better way of, in turn, learning how to prevent them! Good luck ;)

    0 讨论(0)
  • 2020-11-22 17:26

    NilObject's answer is a good start. Here's some supplemental info pertaining to manual memory management (required on the iPhone).

    If you personally alloc/init an object, it comes with a reference count of 1. You are responsible for cleaning up after it when it's no longer needed, either by calling [foo release] or [foo autorelease]. release cleans it up right away, whereas autorelease adds the object to the autorelease pool, which will automatically release it at a later time.

    autorelease is primarily for when you have a method that needs to return the object in question (so you can't manually release it, else you'll be returning a nil object) but you don't want to hold on to it, either.

    If you acquire an object where you did not call alloc/init to get it -- for example:

    foo = [NSString stringWithString:@"hello"];
    

    but you want to hang on to this object, you need to call [foo retain]. Otherwise, it's possible it will get autoreleased and you'll be holding on to a nil reference (as it would in the above stringWithString example). When you no longer need it, call [foo release].

    0 讨论(0)
  • 2020-11-22 17:28

    Lots of good information on cocoadev too:

    • MemoryManagement
    • RulesOfThumb
    0 讨论(0)
提交回复
热议问题