iOS学习(OC语言)知识点整理
一、OC中的内存管理
1)概念:内存管理的对象为所有继承了NSObject的对象,对基本数据(如:int 、float、double...)无效
OC中采用引用计数器对内存做管理,他是一个整数数据,表示对象引用的次数,每个对象分配4字节
的内存空间存放引用计数器当一个对象的引用计数器值为0时,它将被释放,反过来说,如果一个对象的
引用计数器值不为0,这个对象永远不会被释放,除非程序退出。
2) 当给对象发送一个retain消息,引用计数器值会+1,retain消息返回对象本身
3)当给对象发送一个release消息,引用计数器值为-1.
4)给对象发送一个retainCount消息,获取引用计数值
5)当对象的引用计数器值为0,会被销毁,内存空间会被释放
6) 当对象被销毁时,系统自动发送一个dealloc消息
7)一般重写dealloc消息,释放相关资源,一旦重写dealloc方法,一定要调用[super dealloc];
放在最后调用 不能直接调用dealloc方法
8)OC中的内存管理分为:MRC(Manual Reference Counting)手动引用计数
ARC:Automatic Reference Counting:自动引用计数。
9)黄金法则 当使用alloc、new 、copy(mutableCopy)创建一个对象赋值给某个引用时后当不再使用这
个引用时一定要发送release(autoRelease)消息 释放该对象所占用的内存。
10)这里所讲的内存管理为手动内存管理,新建项目是系统默认是自动内存管理的,所以需要手动设置,即在项目
的Build Settings 中找到 Objective-C Automatic Reference Counting 项设置为NO; 同时需要添加对野指
针的监控 即在Edit Scheme 中找到并勾选 Enable Zoombie Objects 项。
11)内存管理操作 实例代码:
1、添加一个Person类 .h文件中不做任何操作
2、在.m 文件中析构系统销毁内存执行方法 例如:
#import "Person.h"
@implementation Person
//析构系统销毁内存执行方法
-(void)dealloc{
NSLog(@"Person dealloc");
}
@end
3、在main文件中执行操作 例如:
Person *p1=[[Person alloc]init];//引用计数器默认+1
[p1 retain]; //引用计数器+1 NSLog(@"retainCount:%ld",p1.retainCount);//结果:2
[p1 release];//引用计数器-1 //retainCount 获取引用计数器的个数
NSLog(@"retainCount:%ld",p1.retainCount);//结果:1
[p1 release];p1=nil;//防止出现野指针,出现不安全因素 OC中可以给空指针对象发送消息
NSLog(@"retainCount:%ld",p1.retainCount);//结果:0
12)在使用组合类是销毁自身对象时需要先销毁组合的对象 ,即在析构方法中销毁组合对象 例如:
//Person 类中有一个Book 类 在Person类的.m文件中添加此方法
-(void)dealloc{
[_book release];
NSLog(@"Person dealloc");
}
13)组合类初始化时需要在父类中添加set方法用于在给组合类赋值时增加引用计数器个数 例如:
//Person 类中Book类的赋值方法
-(void)setBook:(Book*)book{
book=[book retain];
} //Person 类中Book类的取值方法
-(Book*)book{
return _book;
}
14)数组中的内存管理:销毁数组对象时需要先销毁数组中存放的对象 例如:
Book *b0=[[Book alloc]init];
b0.ID=;
NSMutableArray *array=[[NSMutableArray alloc]initWithObjects:b0, nil];
[b0 release];
NSLog(@"%ld",b0.retainCount);
for(int i=;i<;i++){
Book *b=[[Book alloc]init];
b.ID=i+;
NSLog(@"before,book:%ld",b.retainCount);
[array addObject:b];
NSLog(@"after,book:%ld",b.retainCount);
[b release];
NSLog(@"release,book:%ld",b.retainCount);
}
NSLog(@"%ld",array.retainCount);
[array removeLastObject];
NSLog(@"*******");
[array release];
15)数组对象中计数器的运行机制:
1、当用对象创建数组时,数组会自动给对象的引用计数器+1
2、将对象添加到数组时,数组会自动给对象的引用计数器+1
3、将对象从数组中删除时,数组会自动给对象发送release,将引用计数器值-1
4、当数组release时,会自动给所有的元素发送release消息
16)循环引用的内存管理:针对循环引用时我们必须将其中一个对象类型由retain 改为assign 类型
否则将无法彻底释放内存。
17)循环引用时可能会出现头文件重复包含的问题 此时我们将#import 改为 @class 例如: @class Person;
18)autorelease 自动释放池 :将对象放入一个自动释放池中,当自动释放池被销毁时,会给池子中
所有的对象发送 release消息autorelease 方法返回对象本身给对象发送autorelease 消息后引用
计数器值不变
19)创建自动释放池有2种方法:
//方法一
@autoreleasepool {} //方法二
NSAutoreleasePool *pool=[[NSAutoreleasePool alloc]init];
[pool release];
20)autorelease好处:不用担心对象什么时候被释放;缺点:当对象很大时,不能精确控制对象的释放 ;能不用autorelease尽量不用.
21)自动释放池 实例代码:
@autoreleasepool { //自动释放池的开始
Book *b1=[[Book alloc]init];
b1=[b1 autorelease];//返回对象本身
NSLog(@"%ld",b1.retainCount);
b1.ID=;
Book *b5=nil;
@autoreleasepool {
Book *b2=[[[Book alloc]init]autorelease];
b2.ID=;
//[b2 release];//不能,已经有autorelease,在池子被销毁时再次发送release消息,就会出错 //b5=[[[[Book alloc]init]autorelease]autorelease];
//将同一个对象往自动释放池中放入2次,池子被销毁时会发送2次release消息,也会出现野指针错误
b5=[[[Book alloc]init]autorelease];
b5.ID=;
}
//创建自动释放池方法二:
NSAutoreleasePool *pool=[[NSAutoreleasePool alloc]init]; [pool release];
//NSLog(@"id:%d",b5.ID); Person *p1=[Person personWithAge:];
//p1不需要release,p1的创建没有alloc...关键字,已经在类方法中autoreleas过了
//OC中有很多类提供了快速创建实例的类方法,都是autorelease的 NSLog(@"*********");
Student *stu=[Student personWithAge:];
stu.no=; }//自动释放池结束
22)快速构建对象方法 例如:
+(id)personWithAge:(int)newAge
{
// p1对象的释放只能通过autorelease实现
//子类继承后也可以用此方法快速创建子类的对象,需要使用self
//(发送消息的引用,如果是父类就是Person,如果是子类就是Student)
Person *p1=[[[self alloc]init]autorelease];
p1.age=newAge;
return p1;
}