iOS 内存管理

2019-12-03  本文已影响0人  Good_Citizen

先说下我们iOS程序员需要了解的内存四大区域:地址由低到高顺序为代码区,数据段,堆区,栈区

代码区:我们所写的代码编译成二进制就是存储在这个区域

数据段:存放全局变量、静态变量

堆区:通过malloc初始化的对象存储在堆区

栈区:函数中的局部变量都存储在栈区

MRC:手动内存管理

先看下我们平时在MRC下是如何手动管理内存的

#import "WPPerson.h"

int main(int argc,const char* argv[]) {

    @autoreleasepool {

        WPPerson*person = [[WPPerson alloc] init];

        NSLog(@"--%@--",person);

        [person release];

    }

    return0;

}

可以看到,当我们初始化一个对象,不用它的时候需要调用release进行释放,有时会出现提前释放了,这时可以在初始化对象时调用autoreleasefangfa ,这样编译器就会在恰当的时机调用release进行释放

WPPerson*person = [[[WPPerson alloc] init] autorelease];

当出现更复杂的情况,WPPerson对象内部使用到其他对象WPCat,看代码

WPCat有一个对外的方法eat

#import "WPCat.h"

@implementation WPCat

-(void)eat{

    NSLog(@"[WPCat eat]");

}

-(void)dealloc{

    NSLog(@"[WPCat dealloc]");

    [super dealloc];

}

@end

WPPerson有一个成员变量WPCat,并对外提供了setter、getter方法

#import "WPPerson.h"

#import "WPCat.h"

@interface WPPerson()

{

    WPCat*_cat;

}

@end

@implementation WPPerson

-(WPCat*)cat{

    return_cat;

}

-(void)setCat:(WPCat*)cat{

    _cat= cat;

}

-(void)dealloc{

    NSLog(@"[WPPerson dealloc]");

    [super dealloc];

}

@end

mian文件中

#import <Foundation/Foundation.h>

#import "WPPerson.h"

#import "WPCat.h"

int main(int argc,const char* argv[]) {

    @autoreleasepool {

        WPPerson*person = [[WPPerson alloc] init];

        WPCat*cat = [[WPCat alloc] init];

        [person setCat:cat];

        [cat release];

        NSLog(@"%@",[person cat]);

        sleep(2);

        [[person cat] eat];

        [person release];

    }

    return0;

}

初始化WPPerson和WPCat对象,并将cat赋值给person的成员变量cat,person里面的setter方法内部只是简单的赋值操作,而且当赋值操作完成后释放之前的cat对象,这时如果person使用成员变量cat进行操作时会出现EXC_BAD_ACCESS坏内存访问错误,这里睡两秒只是为了体现这个问题,因为当cat调用release方法时,cat对象有可能没有马上被释放,而是加入到了自动释放池中,这时eat方法还是可以调用的,这样就导致cat对象被提前释放了,所以我们一般都会在setter方法中对cat对象进行retain操作,使cat对象的引用计数+1,这样cat对象就不会被销毁

-(void)setCat:(WPCat*)cat{

    _cat= [cat retain];

}

但是这样又会导致cat对象不会被释放,造成内存泄漏,这时只需要在person的dealloc中将成员变量_cat

释放掉即可

-(void)dealloc{

    [_cat release];

    _cat=nil;

    NSLog(@"[WPPerson dealloc]");

    [super dealloc];

}

但如果中途person对象将成员变量重新赋值了一个cat对象,这样就会导致之前的cat对象没有被释放

WPPerson*person = [[WPPerson alloc]i nit];

        WPCat*cat = [[WPCat alloc] init];

        WPCat*cat1 = [[WPCat alloc] init];

        [person setCat:cat];

        [person setCat:cat1];

        [cat release];

        [cat1 release];

        NSLog(@"%@",[person cat]);

        sleep(2);

        [[person cat] eat];

        [person release];

这时只会有一个cat对象被释放,也就是最后赋值的cat1对象,而cat对象没有被释放,这时可以改进setter方法

-(void)setCat:(WPCat*)cat{

    if(_cat!= cat) {

        [_cat release];

        _cat= [cat retain];

    }

}

这就是我们在MRC环境下看到的setter方法写法的由来

上一篇下一篇

猜你喜欢

热点阅读