【iOS】MRC

MRC:手动管理内存。

引用计数

请跳转至引用计数

dealloc方法

dealloc方法是在Objective-C中用来释放对象内存的方法。当一个对象的引用计数变为0时,系统会自动调用该对象的dealloc方法,以释放对象所占用的内存。

一般来说我们不可以直接调用dealloc方法,但是我们可以重写dealloc方法,在其中释放我们自己添加的监听等无法通过ARC来释放的内容。

野指针和空指针

只要一个对象被释放了,我们就称这个对象为「僵尸对象(不能再使用的对象)」。

当一个指针指向一个僵尸对象(不能再使用的对象),我们就称这个指针为「野指针」。
只要给一个野指针发送消息就会报错(EXC_BAD_ACCESS 错误)。

int main(int argc, const char * argv[]) {
    @autoreleasepool {
        NSObject *p = [[NSObject alloc] init]; // 执行完引用计数为 1。
 
        [p release]; // 执行完引用计数为 0,实例对象被释放。
        [p release]; // 此时,p 就变成了野指针,再给野指针 p 发送消息就会报错。
    }
    return 0;
}

为了避免给野指针发送消息会报错,一般情况下,当一个对象被释放后我们会将这个对象的指针设置为空指针。

int main(int argc, const char * argv[]) {
    @autoreleasepool {
        NSObject *p = [[NSObject alloc] init]; // 执行完引用计数为 1。
 
        [p release]; // 执行完引用计数为 0,实例对象被释放。
        p = nil // 将对象指针设为空指针
        [p release]; // 发消息无反应
    }
    return 0;
}

@property参数

在成员变量前加上 @property,系统就会自动帮我们生成基本的 setter / getter 方法,但是不会生成内存管理相关的代码。

@property (nonatomic) int val;

同样如果在 property 后边加上 assign,系统也不会帮我们生成 setter 方法内存管理的代码,仅仅只会生成普通的 getter / setter 方法,默认什么都不写就是 assign。

@property(nonatomic, assign) int val;
/* = 
@property(nonatomic) int val;
*/

如果在 property 后边加上 retain,系统就会自动帮我们生成 getter / setter 方法内存管理的代码,但是仍需要我们自己重写 dealloc 方法。

@property(nonatomic, retain) Room *room;

自动释放池

Objective-C 提供了 autorelease 方法方便我们控制对于对象的释放时机的把握。

autorelease是Objective-C中的一种内存管理机制,可以将一个对象添加到autorelease pool中,以便在稍后的时间内被自动释放。当一个对象被添加到autorelease pool中时,它的引用计数会被增加一次,当pool被释放时,pool中的所有对象会被释放一次。

autorelease 方法会返回对象本身,且调用完 autorelease 方法后,对象的计数器不变。

NSString *str = [[[NSString alloc] initWithString:@"Hello"] autorelease];
NSLog(@"%ld", [str retainCount]); // 1

autorelease原理

autorelease 实际上只是把对 release 的调用延迟了,对于每一个 autorelease,系统只是把该对象放入了当前的 autorelease pool 中,当该 pool 被释放时,该 pool 中的所有对象会被调用 release 方法。

自动释放池是NSAutoreleasePool类的一个实例,它是一个存储autorelease对象的容器。每当一个对象被调用autorelease方法时,它会被添加到最近的自动释放池中。当自动释放池被销毁时,其中的所有对象都会被释放。在Cocoa应用程序中,主循环(RunLoop)会自动创建和释放自动释放池。

autorelease 的注意事项

并不是放到自动释放池代码中,都会自动加入到自动释放池。

@autoreleasepool {
    // 因为没有调用 autorelease 方法,所以对象没有加入到自动释放池
    Person *p = [[Person alloc] init];
    [p run];
}

autorelease 是一个方法,只有在自动释放池中调用才有效。

@autoreleasepool {
}
// 没有与之对应的自动释放池, 只有在自动释放池中调用autorelease才会放到释放池
Person *p = [[[Person alloc] init] autorelease];
[p run];
 
// 正确写法
@autoreleasepool {
    Person *p = [[[Person alloc] init] autorelease];
 }
 
// 正确写法
Person *p = [[Person alloc] init];
@autoreleasepool {
    [p autorelease];
}

autorelease 误区

不要连续调用 autorelease。

@autoreleasepool {
 // 错误写法, 过度释放
    Person *p = [[[[Person alloc] init] autorelease] autorelease];
 }

调用 autorelease 后又调用 release(错误)。

@autoreleasepool {
    Person *p = [[[Person alloc] init] autorelease];
    [p release]; // 错误写法, 过度释放
}

你可能感兴趣的:(ios,objective-c,xcode)