Objective-C Memory Management: crash on release

前端 未结 5 1360
情深已故
情深已故 2021-01-29 00:07

I\'m new to Objective-C and can\'t seem to get the memory management code right. I have the following code:

Media* myMedia = [self.myMediaManager getNextMedia];
         


        
相关标签:
5条回答
  • 2021-01-29 00:13

    The general rule for memory management is as follows:

    For every retain, alloc, copy, or new, you need to call release or autorelease.

    Since you did not call any these, you do not need to release myMedia.

    For more information, take a look at this other answer I posted that deals with the subject. Also, since you are new to iOS development, I suggest looking at this answer as well.

    0 讨论(0)
  • 2021-01-29 00:14

    At this point, since you are just learning, you should probably just start off using ARC with the iOS5 beta versions of XCode. It's good to have an understanding but using ARC will save you many potential pitfalls - by the time you learn enough to produce something iOS5 will be out. You can still build applications targeting iOS4, so you'll still be able to reach a lot of people.

    0 讨论(0)
  • 2021-01-29 00:14

    This updated code is suspicious:

    Media* nextMedia = [[Media alloc] init];
    
    [self setNextMediaIndex];
    
    if (self.mediaIndex > -1)
    {
        nextMedia = [mediaArray objectAtIndex: self.mediaIndex];
    }
    

    Depending on the condition in the if() clause, you assign a new value to nextMedia, which makes the value you just allocated unreachable, i.e. it can't be released.

    Also, you don't retain the value you get from the array, so you should not release it either. But if the if() clause does not run, you still have the instance you alloc-ed, and that should be released.

    That is not good. Try:

    Media* nextMedia = [[Media alloc] init];
    [self setNextMediaIndex];
    if (self.mediaIndex > -1)
    {
        [nextMedia release];
        nextMedia = [[mediaArray objectAtIndex: self.mediaIndex] retain];
    }
    

    You could also do (and I would prefer that):

    Media *nextMedia;
    [self setNextMediaIndex];
    if (self.mediaIndex > -1)
    {
        nextMedia = [[mediaArray objectAtIndex: self.mediaIndex] retain];
    }
    else
    {
        nextMedia = [[Media alloc] init];
    }
    

    Now you can release nextMedia when that is not needed anymore, without any ambiguity about the retain count.

    0 讨论(0)
  • 2021-01-29 00:19

    Since myMedia is not retained here, you don't need to release it. When the origin (self.myMediaManager) releases it, it gets destroyed immediately.

    NSString *string = [[NSString alloc] init];
    [string release]; // now we have to release the string, since we allocated it.
    
    NSString *string = self.navigationItem.title;
    // now we don't, since it's a property of `navigationItem` and we didn't retain it.
    
    0 讨论(0)
  • 2021-01-29 00:20

    Only objects that you own can be released.

    You can release objects if you new, alloc, copy, mutableCopy or retain them first.

    Since there is no alloc/copy/retain in [self.myMediaManager getNextMedia]; you can't release it.

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