1. Object-C非正式协议与正式协议的区别
- 所谓的非正式协议就是类别,即凡是NSObject或其子类的类别,都是非正式协议
- 正式协议从概念上理解起来就简单的多了,它指的是一个以@protocol方式命名的方法列表,与非正式协议相比不同的是,它要求显示的采用协议
- 正式协议可以将业务中的方法定义剥离出来,形成一个单独的文件
2. 类别与类扩展的区别
- 类别中只能增加方法
- 类扩展不仅可以增加方法,还可以增加实例变量(或者合成属性),只是该实例变量默认是@private类型的(作用范围只能在自身类,而不是子类或其他地方)
- 类扩展中声明的方法没被实现,编译器会报警,但是类别中的方法没被实现编译器是不会有任何警告的。这是因为
类扩展是在编译阶段被添加到类中,而类别是在运行时添加到类中
- 类扩展不能像类别那样拥有独立的实现部分(@implementation部分),也就是说,
类扩展所声明的方法必须依托对应类的实现部分来实现
- 定义在 .m 文件中的类扩展方法为私有的,定义在 .h 文件(头文件)中的类扩展方法为公有的。类扩展是在 .m 文件中声明私有方法的非常好的方式
3. 内存管理
管理范围;
任何继承NSObject的对象,对其他的基本数据类型无效
僵尸对象
- 指向已经删除的对象内存空间
- 已经被销毁的对象(不能再使用的对象)
简而言之,就是过度释放的对象
野指针
指向僵尸对象(不可用内存)的指针
给野指针发消息会报EXC_BAD_ACCESS
错误空指针和野指针的区别
- 没有存储任何内存地址的指针就称为空指针(NULL或 nil指针)
避免使用僵尸对象的方法是,
在堆内存释放后,给对象赋值为nil,因为给nil发送消息是没有任何效果的。
dealloc方法
dealloc方法用来释放这个对象所占的内存(包括成员变量)和其它资源。
不要使用dealloc方法来管理稀缺资源,比如文件,网络链接等。因为由于bug或者程序意外退出,dealloc方法不能保证一定会被调用
Accessor Methods和内存管理
如果getter返回的对象已经被运行环境回收了,那么这个getter的返回值是毫无意义的。这就需要在setter方法里“拥有”相应的property
setter方法需要保证对这个成员变量的“拥有”: -(void)setCount:(NSNumber *)newCount { [newCount retain]; //拥有新值 [_count release]; //放弃老值 _count = newCount; //简单赋值 }