iOS 内存管理

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

©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

  • 1. 内总管理原则(引用计数) IOS的对象都继承于NSObject, 该对象有一个方法:retainCount...
    lilinjianshu阅读 2,254评论 0 2
  • 前言 本篇博客手动 copy 自 iOS 开发:彻底理解 iOS 内存管理(MRC、ARC)。谢谢原作者的详细分析...
    四月_Hsu阅读 780评论 0 2
  • 内存管理方式 首先明确一点,无论在MRC还是ARC情况下,Objective-C采用的是引用计数式的内存管理方式,...
    软件iOS开发阅读 1,771评论 0 9
  • 软件运行时会分配和使用设备的内存资源,因此,在软件开发的过程中,需要进行内存管理,以保证高效、快速的分配内存,并且...
    maTianHong阅读 797评论 0 0
  • 朋友们说 下雪了 我也是从手机的微信里看到的 其实 窗外的雪早就纷纷扬扬了 厚厚的窗帘 明亮的灯光 温...
    佛澄心阅读 200评论 0 2

友情链接更多精彩内容