OC内存管理和内存管理原则

转载 2015年11月18日 19:31:52

内存管理

    范围:

  • 任何继承了NSObject 的对象,对基本数据类型无效

   原理:

  • 每个对象内部都保存了一个与之相关联的整数,称为引用计数器(auto reference count)
  • 每当使用 allocnew或者copy创建一个对象时,对象的引用计数器被设置为1
  • 给对象发送一条retain消息(即调用retain方法),可以使引用计数器值+1
  • 给对象发送一条release消息,可以使引用计数器值-1
  • 当一个对象的引用计数器值为0时,那么它将被销毁,其占用的内存被系统回收,OC也会自动向对象发送一条dealloc消息。一般会重写dealloc方法,在这里释放相关资源。一定不要直接调用dealloc方法
  • 可以给对象发送retainCount消息获得当前的引用计数器值。

内存管理原则

  1. 谁创建,谁释放(“谁污染,谁治理”)。如果你通过allocnew或者(mutable)copy来创建一个对象,那么你必须调用releaseautorelease。或句话说,不是你创建的,就不用你去释放
  2. 一般来说,除了allocnewcopy之外的方法创建的对象都被声明了autorelease(autorelease是延迟释放内存,不用你自己去手动释放,系统会知道在什么时候该去释放掉它。)
  3. retain,谁release。只要你调用了retain,无论这个对象是如何生成的,你都要调用release

下面是测试代码:

新建一个基于“Command Line Tool”的工程,名为“内存管理1-retain和release的简单使用”,再新建一个Student类。

Student.h

  1. //  
  2. //  Student.h  
  3. //  内存管理1-retain和release的简单使用  
  4. //  
  5.   
  6. #import <Foundation/Foundation.h>  
  7.   
  8. @interface Student : NSObject  
  9. {  
  10.     int age;  
  11. }  
  12.   
  13. @property int age;  
  14.   
  15. @end  

Student.m

  1. //  
  2. //  Student.m  
  3. //  内存管理1-retain和release的简单使用  
  4.   
  5. #import "Student.h"  
  6.   
  7. @implementation Student  
  8.   
  9. @synthesize age;  
  10.   
  11. //重写dealloc方法,当引用计数器(auto reference count)为零的时候调用,注意,没有在.h头文件中声明的方法都属于私有方法。  
  12. - (void)dealloc{  
  13.       
  14.     //insert your code here...  
  15.     NSLog(@"%@被销毁了",self);  
  16.       
  17.     [super dealloc];//一定要调用super的dealloc方法,而且最好放在最后面实现  
  18. }  
  19.   
  20. @end  

main.m

  1. //  
  2. //  main.m  
  3. //  内存管理1-retain和release的简单使用  
  4. //
  5.   
  6. #import <Foundation/Foundation.h>  
  7. #import "Student.h"  
  8.   
  9. int main(int argc, const char * argv[])  
  10. {  
  11.   
  12.     @autoreleasepool {  
  13.           
  14.         Student *stu = [[Student alloc]init];//alloc一次,引用计数器为1  
  15.         //Student *stu = [[[Studnet alloc]init] autorelease]; //这样写的话系统会在适当的地方对stu的内存进行自动回收,就不用自己写release回收了  
  16.           
  17.         //z代表无符号  
  18.         NSLog(@"count:%zi", [stu retainCount]);  
  19.           
  20.         [stu retain];//引用计数器变为2  
  21.         NSLog(@"count:%zi", [stu retainCount]);  
  22.           
  23.         [stu release];//引用计数器变为1  
  24.         NSLog(@"count:%zi", [stu retainCount]);          
  25.           
  26.         [stu release];//release一次,引用计数器减1,变为0,,,然后会调用dealloc方法  
  27.           
  28.     }  
  29.     return 0;  
  30. }  

运行结果:

2013-08-26 10:53:14.506内存管理1-retainrelease的简单使用[754:303] count:1

2013-08-26 10:53:14.508内存管理1-retainrelease的简单使用[754:303] count:2

2013-08-26 10:53:14.509内存管理1-retainrelease的简单使用[754:303] count:1

2013-08-26 10:53:14.509内存管理1-retainrelease的简单使用[754:303] <Student: 0x100109a80>被销毁了


1.在实际开发中很少会用到new,一般创建对象咱们看到的全是[[className alloc] init]

但是并不意味着你不会接触到new,在一些代码中还是会看到[className new],

还有去面试的时候,也很可能被问到这个问题。

2.那么,他们两者之间到底有什么区别呢

我们看源码:

  1. new 
  2. id newObject = (*_alloc)((Class)self, 0); 
  3. Class metaClass = self->isa; 
  4. if (class_getVersion(metaClass) > 1) 
  5. return [newObject init]; 
  6. else 
  7. return newObject; 
  8.  
  9. //而 alloc/init 像这样: 
  10. + alloc 
  11. return (*_zoneAlloc)((Class)self, 0, malloc_default_zone());  
  12. - init 
  13. return self; 

通过源码中我们发现,[className new]基本等同于[[className alloc] init];

区别只在于alloc分配内存的时候使用了zone.

这个zone是个什么东东呢?

它是给对象分配内存的时候,把关联的对象分配到一个相邻的内存区域内,以便于调用时消耗很少的代价,提升了程序处理速度;

3.而为什么不推荐使用new?

不知大家发现了没有:如果使用new的话,初始化方法被固定死只能调用init.

而你想调用initXXX怎么办?没门儿!据说最初的设计是完全借鉴Smalltalk语法来的。

传说那个时候已经有allocFromZone:这个方法,

但是这个方法需要传个参数id myCompanion = [[TheClass allocFromZone:[self zone]] init];

这个方法像下面这样:

  1. + allocFromZone:(void *) z 
  2. return (*_zoneAlloc)((Class)self, 0, z);  
  3.  
  4. //后来简化为下面这个: 
  5. + alloc 
  6. return (*_zoneAlloc)((Class)self, 0, malloc_default_zone());  

但是,出现个问题:这个方法只是给对象分配了内存,并没有初始化实例变量。

是不是又回到new那样的处理方式:在方法内部隐式调用init方法呢?

后来发现“显示调用总比隐式调用要好”,所以后来就把两个方法分开了。

概括来说,new和alloc/init在功能上几乎是一致的,分配内存并完成初始化。

差别在于,采用new的方式只能采用默认的init方法完成初始化,

采用alloc的方式可以用其他定制的初始化方法。

iOS 内存管理基本原则

我们知道objc中创建对象是存放在堆中的(基本数据类型除外,是由系统自己管理,并存放在栈中),系统不会自动释放堆中的内存。如果创建完的对象存放在堆中后并使用完没有得到及时的释放,会占用的内存。但是ob...
  • aaaa99996666
  • aaaa99996666
  • 2016年05月07日 19:33
  • 1735

内存管理基本原则

之前:OC内存管理遵循“谁创建,谁释放,谁引用,谁管理”的机制,当创建或引用一个对象的时候,需要向她发送alloc、copy、retain消息,当释放该对象时需要发送release消息,当对象引用计数...
  • u012701023
  • u012701023
  • 2016年03月24日 15:41
  • 828

关于OC中的Block使用以及ARC和MAR下的内存管理方式

1.什么是Block?     Block是属于C语言框架     Block是一种数据类型(类似int)     Block是一段代码块,只有在被调用的时候被执行(类似函数和方法) ...
  • XieEXiaoZhang
  • XieEXiaoZhang
  • 2016年07月25日 15:19
  • 987

Swift OC 内存管理

???? 1.??????? ???????,???????????,?????????????????alloc?free,?????????????????????,????,????????...
  • JoeTongOwn
  • JoeTongOwn
  • 2016年08月02日 15:01
  • 143

对iOS开发中内存管理的一点总结与理解

做iOS开发也已经有两年的时间,觉得有必要沉下心去整理一些东西了,特别是一些基础的东西,虽然现在有ARC这种东西,但是我一直也没有去用过,个人觉得对内存操作的理解是衡量一个程序员成熟与否的一个标准。好...
  • ios_che
  • ios_che
  • 2013年11月05日 17:31
  • 7006

关于iOS内存管理的规则思考

关于iOS内存管理的规则思考 自己生成的生成的对象,自己持有。 非自己生成的对象,自己也能持有。 不在需要自己持有的对象时释放。 非自己持有的对象无法释放。 注:这里的自己是对象使用的环境,理解为编程...
  • wxs0124
  • wxs0124
  • 2016年12月21日 17:08
  • 2901

[读书笔记]iOS与OS X多线程和内存管理 [GCD部分]

3.2 GCD的API 苹果对GCD的说明:开发者要做的只是定义想执行的任务并追加到适当的Dispatch Queue中。 “Dispatch Queue”是执行处理的等待队列。通过dis...
  • zyb050607
  • zyb050607
  • 2015年03月13日 10:15
  • 1330

Objective-C 内存管理机制

一、Objective-C内存管理的对象。二、为什么要使用内存管理。三、Objective-C管理内存的方式。四、与内存有关的修饰符。五、MRC与ARC混编...
  • lover0920
  • lover0920
  • 2016年01月05日 08:57
  • 1658

OC学习篇之---内存管理介绍和使用

在之前的一片文章我们说了OC中谓词操作:http://blog.csdn.net/jiangwei0910410003/article/details/41923507,从今天开始我们就来看一下OC中...
  • jiangwei0910410003
  • jiangwei0910410003
  • 2014年12月14日 16:16
  • 9017

OC的手动内存管理基础(IOS开发)

OC的内存管理分为三种:          1,手动内存管理(MRC).          2,自动内存管理(ARC)          3,垃圾回收      IOS系统并不支持垃圾回收,现在主要使...
  • aa605397965
  • aa605397965
  • 2015年07月13日 23:39
  • 768
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:OC内存管理和内存管理原则
举报原因:
原因补充:

(最多只允许输入30个字)