本篇博文,将给大家介绍下再Objective-C中如何使用内存管理。一个程序运行的时候,如果不及时的释放没有用的空间内存。那么,程序会越来越臃肿,内存占用量会不断升高。我们在使用的时候,就会感觉很卡,最终使得程序运行奔溃。因此,将无效的内存及时清理释放,是非常有必要的。

  一个对象在最初创建使用,到最后的回收释放,经历的是怎样一个过程呢?
包括:诞生(通过alloc或new方法实现)、生存(接收消息并执行操作)、交友(通过复合以及向方法传递参数)、最终死去(被释放掉)。

一、引用计数

  在对象创建的时候,Cocoa使用了一种叫引用计数的技术:
1)当一个对象被访问的时候,引用计数器的值就加1,可以给对象发送一条retain消息
2)当结束该对象的访问的时候,引用计数器的值就减1,可以给对象发送一条release消息
3)当引用计数器的值为0的时候,表示不再访问该对象,则其占用的系统内存将被回收重用,会自动给对象发送一条dealloc消息,一般都会重写dealloc方法;
4)要获得保留计数器当前的值,可以发送retainCount消息

  下面,介绍下几种方法的声明和实现方法:

首先,新建一个RetainTracker的类,修改类的声明文件和实现方法:

 // RetainTracker.h

 #import <Foundation/Foundation.h>

 @interface RetainTracker : NSObject
-(id) retain;
-(oneway void)release;
-(NSUInteger)retainCount;
@end
 // RetainTracker.m

 #import "RetainTracker.h"

 @implementation RetainTracker
-(id) init
{
if(self == [super init])
{
NSLog(@"init: Retain count of %lu.", [self retainCount]);
}
return (self);
} -(void) dealloc
{
NSLog(@"dealloc called. ByeBye!");
[super dealloc];
} @end

然后在main.m主函数中调用retain,release,retainCount,dealloc等方法:

 #import <Foundation/Foundation.h>
#import "RetainTracker.h" int main(int argc, const char * argv[])
{
RetainTracker *tracker = [RetainTracker new];// count =1 [tracker retain];
NSLog(@"retainCount: %lu", [tracker retainCount]);// count =2 [tracker retain];
NSLog(@"retainCount: %lu", [tracker retainCount]);// count =3 [tracker release];
NSLog(@"retainCount: %lu", [tracker retainCount]);// count =2 [tracker release];
NSLog(@"retainCount: %lu", [tracker retainCount]);// count =1 [tracker retain];
NSLog(@"retainCount: %lu", [tracker retainCount]);// count =2 [tracker release];
NSLog(@"retainCount: %lu", [tracker retainCount]);// count =1 [tracker release];// count =0, dealloc return ();
}

运行结果如下:

二、自动释放

  大家都知道,当对象不再使用的时候,要及时释放。但是在某些情况下,弄清楚什么时候不再使用一个对象并不容易。如果能够自动释放就好了。很幸运,Cocoa中有一个自动释放池(autorelease pool)。细心的朋友可以发现,在ios5以后,每次新建项目,在main函数中都有个@autoreleasepool方法,这就是将执行的代码都加到自动释放池中。

  NSObject类提供了一个叫做autorelease的方法:

 -(id) autorelease;

  该方法预先设定一条会在未来某个时间发送的release消息。当给一个对象发送autorelease消息的时候,实际上是将该对象添加到自动释放池中。当自动释放池被销毁时,会向该池中的所有对象都发送release消息。

  那么,接下来,我们就用添加到自动释放池的方法来修改上面的例子。RetainCount类的内容不改变,只要修改main主函数中的内容:

 /* use auto release pool */
int main(int argc, const char * argv[])
{
NSAutoreleasePool *pool;
pool = [[NSAutoreleasePool alloc] init]; RetainTracker *tracker = [RetainTracker new];// count =1
NSLog(@"after new, tracker: %lu", [tracker retainCount]);// count =1 [tracker retain];
NSLog(@"after retain, tracker: %lu", [tracker retainCount]);// count =2
[tracker autorelease];
NSLog(@"after autorelease, tracker: %lu", [tracker retainCount]);// count =2 [tracker release];
NSLog(@"after release, tracker: %lu", [tracker retainCount]);// count =1
NSLog(@"releasing pool");
[pool release]; // 销毁自动释放池 @autoreleasepool {
RetainTracker *tracker2;
tracker2 = [RetainTracker new]; // count = 1
[tracker2 retain]; //count =2
[tracker2 autorelease]; // count still = 2
[tracker2 release]; //count = 1
NSLog(@"auto releasing pool.");
} return ();
}

  运行结果:

  tracker对象,通过autorelease消息,将该对象添加到自动释放池中。当pool自动释放池发送release消息的时候,pool对象的引用计数器的值为0,则该自动释放池要被销毁,其dealloc方法被调用。使得自动释放池中的对象也都跟随其一起被销毁。

三、内存管理规则

  接下来,就给大家介绍下Cocoa的内存管理的几个规则。

1). 当你使用 new, alloc, copy 方法创建一个对象时,该对象保留计数器的值为1;
     当不使用的时候,要发送一条release或autorelease消息,销毁对象。

 NSMutableArray *array;
array = [[NSMutableArray alloc] init]; //count =1
//use the array
[array release];//dealloc, count =0

2). 当你通过别的方法获得一个对象时,假设该对象的保留计数器的值为1,而已经被设置为自动释放,则不需要执行任何操作来清理该对象。

 NSMutableArray *array;
array = [NSMutableArray arrayWithCapacity: ]; //count =1, autorelease;
//use the array

3). 自动释放池销毁的时间是完全确定的,它在循环阶段是不会被销毁的。如果一个循环要添加到自动释放池中的对象很多的时候,可以考虑循环一部分后先分批释放掉一些,然后再创建新的自动释放池。这样就保证自动释放池的分配和销毁操作代价尽可能的小。

 NSAutoreleasePool *pool;
pool = [ [NSAutoreleasePool alloc] init ];
int i;
for(i=; i<; i++)
{
id object = [someArray objectAtIndex: i];
NSString *desc = [object descrption];
If(i% == )
{
// 每循环一百次就清空一次,然后新建一个自动释放池
[pool release];
pool = [[NSAutoreleaasePool alloc]init];
}
}
[pool release];

4). 自动引用计数(Auto Reference Counting,即:ARC),如果你启用了ARC,只要像平时一样按需分配并使用对象,编译器会帮你插入retain和release,无需你自己手动添加。ARC只能保留Objective-c的指针对象,即:继承NSObject的对象。

5). Ios 5 以上,有了归零弱引用,因为在指向的对象释放之后,这些弱引用就会被设置为零(即:nil),然后对象就会像平常指向nil值的指针一样被处理。使用前要先明确声明,声明方法如下:

 _weak NSString *myString; 或
@property(weak) NSString *myString;

《objective-c基础教程》学习笔记(十)—— 内存管理的更多相关文章

  1. linux kernel学习笔记-5内存管理_转

    void * kmalloc(size_t size, gfp_t gfp_mask); kmalloc()第一个参数是要分配的块的大小,第一个参数为分配标志,用于控制kmalloc()的行为. km ...

  2. XV6学习笔记(2) :内存管理

    XV6学习笔记(2) :内存管理 在学习笔记1中,完成了对于pc启动和加载的过程.目前已经可以开始在c语言代码中运行了,而当前已经开启了分页模式,不过是两个4mb的大的内存页,而没有开启小的内存页.接 ...

  3. Python基础教程学习笔记:第一章 基础知识

    Python基础教程 第二版 学习笔记 1.python的每一个语句的后面可以添加分号也可以不添加分号:在一行有多条语句的时候,必须使用分号加以区分 2.查看Python版本号,在Dos窗口中输入“p ...

  4. COCOS学习笔记--Cocod2dx内存管理(三)-Coco2d-x内存执行原理

    通过上两篇博客.我们对Cocos引用计数和Ref类.PoolManager类以及AutoreleasePool类已有所了解,那么接下来就通过举栗子来进一步看看Coco2d-x内存执行原理是如何的. / ...

  5. 嵌入式linux学习笔记1—内存管理MMU之虚拟地址到物理地址的转化

    一.内存管理基本知识 1.S3C2440最多会用到两级页表:以段的方式进行转换时只用到一级页表,以页的方式进行转换时用到两级页表.页的大小有三种:大页(64KB),小页(4KB),极小页(1KB).条 ...

  6. Linux内核学习笔记——内核内存管理方式

    一 页 内核把物理页作为内存管理的基本单位:内存管理单元(MMU)把虚拟地址转换为物理 地址,通常以页为单位进行处理.MMU以页大小为单位来管理系统中的也表. 32位系统:页大小4KB 64位系统:页 ...

  7. ios学习笔记之内存管理

    一,内存管理类型定义      1,基本类型  任何C的类型,eg:      int,short,char,long,long long,struct,enum,union等属于基本类型或结构体   ...

  8. Cocos2D-X2.2.3学习笔记3(内存管理)

    本章节介绍例如以下: 1.C/C++内存管理机制 2.引用计数机制 3.自己主动释放机制 1.C/C++内存管理机制 相信仅仅要懂oop的都知道NEW这个keyword吧,这个通俗点说事实上就是创建对 ...

  9. arm-linux学习笔记3-linux内存管理与文件操作

    配置好linux系统之后需要vim配置一下,有助于我们的编程,主要的配置如下 在/etc/vim/vimrc文件中 "显示行号 set number "自动缩进 set autoi ...

  10. 《C#高级编程》学习笔记----c#内存管理--栈VS堆

    本文转载自Netprawn,原文英文版地址 尽管在.net framework中我们不太需要关注内存管理和垃圾回收这方面的问题,但是出于提高我们应用程序性能的目的,在我们的脑子里还是需要有这方面的意识 ...

随机推荐

  1. maven异常

    1.There are test failures pom中加入: <build> <plugins> <plugin> <groupId>org.ap ...

  2. hdu - 3959 Board Game Dice(数学)

    这道题比赛中没做出来,赛后搞了好久才出来的,严重暴露的我薄弱的数学功底, 这道题要推公式的,,,有类似于1*a+2*a^2+3*a^3+...+n*a^n的数列求和. 最后画了一张纸才把最后的结果推出 ...

  3. 重新理解JS的6种继承方式

    写在前面 一直不喜欢JS的OOP,在学习阶段好像也用不到,总觉得JS的OOP不伦不类的,可能是因为先接触了Java,所以对JS的OO部分有些抵触. 偏见归偏见,既然面试官问到了JS的OOP,那么说明这 ...

  4. vim安装YouCompleteMe 插件

    要安装YouCompleteMe ,vim须支持python.看是否支持,可以在vim中:version 查看, 如果python前有+号,就是支持,减号就是不支持. 如果不支持,需要以编译安装方式重 ...

  5. 从源代码分析Universal-Image-Loader中的线程池

    一般来讲一个网络访问就需要App创建一个线程来执行,但是这也导致了当网络访问比较多的情况下,线程的数目可能积聚增多,虽然Android系统理论上说可以创建无数个线程,但是某一时间段,线程数的急剧增加可 ...

  6. [JS2] JS是弱类型

    <html> <head> <title>JavaScript 是弱类型的</title> <Script Language="Java ...

  7. django上传文件

    template html(模板文件): <form enctype="multipart/form-data" method="POST" action ...

  8. IMP-00038:无法转换为环境字符集句柄

    参考解决方案:http://www.cnblogs.com/wangsaiming/p/4947151.html

  9. GIt 从入门到放弃

    一.注册github账号 github网址https://github.com/ 下一步 然后,你的邮箱会收到一封邮件,进行邮箱验证就行了 二.创建github仓库 登录你的github,在首页的右方 ...

  10. 技巧~向URL地址添加参数

    只是个小技巧,感觉挺有用,拿出来分享一下 功能:在URL地址上添加参数,如果只有一个,会是?userid=1,需要是大于1个,会是&userid=1 实现: private string Ad ...