美文网首页
iOS 内存管理

iOS 内存管理

作者: Good_Citizen | 来源:发表于2019-12-03 17:58 被阅读0次

    先说下我们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方法写法的由来

    相关文章

      网友评论

          本文标题:iOS 内存管理

          本文链接:https://www.haomeiwen.com/subject/mokcgctx.html