day15
1.SEL
感觉就是指针函数
2.MRC
OC对象内存管理的法则:
![Uploading class_256917.png . . .]
- 有加就有减
- 一次alloc对应一次release
- 一次new对应一次release
- 一次reatin对应一次release
MRC与ARC
ARC: Automatic Reference Counting 自动引用计数
不需要程序员管理对象的内存
Xcode在编译代码时, 会自动的在合适的位置添加retain/reslease等代码
从Xcode4.x之后, 只要创建一个项目, 默认就是ARC的, 所以默认就不能使用retain等方法MRC: Manul Reference Counting 手动引用计数
需要程序员管理对象的内存
所有通过alloc/new等方法创建出来的对象, 都需要我们自己管理它的计数器
与MRC有关的4个方法
retain方法 : 给对象发送一个retain消息, 那么对象的引用计数器就会+1,创建对象时引用计数器初始化为1.
release方法 : 给对象发送一个release消息, 那么对象的计数器就会-1,当引用计数器为0时,对象将会被释放.
retainCount方法 : 获得引用计数器的值.(这个值可能将不准确,所以基本无用)
dealloc方法 : 当对象销毁时,系统将自动调用该函数,该方法经常被重写,一般在这个方法中release对象创建时生成的对象(对象类型的成员变量),重写该方法时必须在结束时调用父类的dealloc方法([super dealloc]),就像在重写init(构造方法)时必须先调用父类的init方法一样.
dealloc方法
当对象销毁时,系统将自动调用该函数,
- 重写dealloc方法时必须在结束时调用父类的dealloc方法
- (void)dealloc {
NSLog(@"Person对象被释放了");
// 注意: 如果重写了dealloc方法, 那么必须再重写的方法中调用[super dealloc];
// [super dealloc]一定要写在所有代码的最后面
[super dealloc];
}
- 重写dealloc方法时使用
self.userName = nil;
比使用[_userPwd release];
更好,这样可以防止因过度释放引起的野指针错误(访问已经销毁了的对象)
- (void)dealloc {
NSLog(@"%s", __func__);
// 只会将对象释放, 不会将对象设置nil
// [_userPwd release];
// [_userName release];
// 牛逼的写法, 它会将对象释放并且设置为nil
self.userPwd = nil; // [self setUserPwd: nil];
self.userName = nil;
[super dealloc];
}
对象类型set方法的写法
- (void)setRoom:(Room *)room {
if (_room != room) {
// 释放以前的房子
[_room release]; // [nil release];
// 保存现在的房子
_room = room;
[_room retain];
}
}
使用@property(nonatomic, retain)Room * room;
来声明的属性将会被编译器翻译成以上代码.
3.property修饰符
-
修饰是否需要对传入的形参进行retain, 计数器是否需要+1
- retain用于对象类型
- assign用于基本数据类型
-
修饰是否需要重新指定setter/getter方法名
- getter=
- setter=
-
修饰是否需要生成setter/getter方法
- readonly
- readwrite
-
在iOS开发中, 99%的情况都写nonatomic
- nonatomic : 非原子性(多线程使用)
- atomic: 原子性
注意: 不同类型的修饰符可以结合在一起使用, 多个修饰符之间使用,隔开
4.循环retain
什么是循环retain?
- A类retain了B类
- B类又retain了A类
循环retain的弊端会导致对象无法正常释放
如何解决循环retain呢?
如果到A类需要用到B类, 而B类又需要用到A类时,
也就是说如果A类需要保存B类, B类又需要保存A类时,
我们可以让一方retain(强指向), 一方不retain(弱指向),
值得注意的是不管是MRC,还是ARC,循环retain都会发生
5.@class
- @class的作用:
在.h文件中使用@class来代替#import,在.m文件中再使用#import
这样可以大量减少编译时的拷贝.
- 什么时候拷贝?
.h文件只要有一点改动,引用该.h文件的其它.h文件都会从新拷贝.
而这些从新拷贝的.h文件,也是被其他的文件引用了(.h文件就是给其它.h.m文件引用的),这样就会发生重新拷贝. (连锁重新拷贝)
而在.h使用@class, 在.m中使用#import. 就不会发生连锁从新拷贝了,因为 .h文件改变了,只有引用它的.m文件重新拷贝,但.m文件不会被其他文件引用.所以就不会在发生重新拷贝了.