一、分类(Category)
1.你都用分类在项目中做了哪些事情?
- 声明私有方法。
- 分解体积庞大的类文件。
- 把Framework 的私有方法公开。
2.特点: - 运行时决议。
- 可以为系统类添加分类。
3.分类中都可以添加哪些内容?
- 实例方法。
- 类方法。
- 协议
- 属性(只是能手动添加get、setter方法,不能直接添加实例属性。若非要添加的话可以使用“关联对象”的技术)
4.加载调用栈
这里的“images”为镜像
_objc_init -> map_2_images ->map_images_nolock ->_read_images ->remethodizeClass
分类总结:
- 分类添加的方法可以“覆盖”原类方法。
- 同名分类方法谁能生效取决于编译顺序。
- 名字相同的分类会引起编译报错。
二.扩展(Extension)
1.一般扩展做什么?
- 声明私有属性
- 声明私有方法
- 声明私有成员变量
分类与扩展的区别?
区别:
- 扩展是编译时决议,分类是运行时决议
- 分类有声明,有实现。而扩展 只是以声明的形式存在,多数情况下寄生于宿主类的.m中
- 不能为系统类添加扩展。
三.代理(Delegate)
- 准确的说是一种软件设计模式
- iOS 当中以@protocol 形式提现
- 传递方式一对一,(而通知是一对多).
- 一般声明为weak以规避循环引用。
- 代理主要由三部分组成:
协议:用来约束代理双方可以做什么,必须做什么。
代理:根据协议,完成委托方需要实现的功能(方法)。
委托:根据协议,指定代理去完成什么功能。
四.通知
特点:
- 是使用观察者模式来实现的用于夸层传递消息的机制。
- 传递方式为:一对多。
- 如何实现通知机制?
通知中心这个系统类中会可能会维护一个Notification_Map的表或者而说字典,这个key也就是notificationName ,value 就是Observers_List是一个数组列表,数组列表中的每一个成员首先包含接受通知的观察者,其次关于观察者调用的方法,每一个元素中也包含通知调用的回调方法。
五.KVO
- KVO 是key-value observing的缩写。
- KVO是Objective-C对观察者设计模式的又一实现。
- Apple 使用了isa混写(isa-swizzling)来实现KVO。
isa混写在KVO试怎么实现的呢?工作原理是什么?
当我们调用了isa pointer 方法的时候,系统会动态创建一个NSKVONotifying_A这样一个类,同时将原来类的isa指针指向一个新创建的NSKVONotifying_A类。NSKVONotifying_A是原来类的一个子类,两者是继承关系,目的是为了重写原有类的Setter方法,重写的setter方法负责通知所有观察对象。
系统重写的两个方法如下:
- (void)willChangeValueForKey:(NSString *)key
- (void)didChangeValueForKey:(NSString *)key
总结: - 使用setter方法改变值KVO才会生效。
- 使用setValue:forKey:改变KVO才会生效。
- 成员变量直接修改需手动添加KVO才会生效。
六.KVC
- 键值编码(Key-Value Coding),它是一种通过key值访问类属性的机制,而不是通过setter/getter方法访问。
- 实现的原理:
1.设置值的实现原理
(1).首先搜索是否有setKey:的方法,没有则会搜索是否有setIsKey:的方法。
(2).如果经过搜索没有找到setKey:的方法,此时就会查找+ (BOOL)accessInstanceVariablesDirectly;(是否直接访问成员变量)
这个方法,若该方法返回NO,则直接调用- (nullable id)valueForUndefinedKey:;(默认是抛出异常)
。若返回YES,会按照_key
、_iskey
、key
、iskey
的顺序搜索成员名。
(3).若在第二步还没搜索到的话就会调用- (nullable id)valueForUndefinedKey:方法
。
2.取值的实现原理
(1).按照getKey
、key
、isKey
、_getKey
、_key
这样的一个顺序依次去查找并获取值,若直接获取到则返回,后续方法不再执行。如果是BOOL或者Int等值类型, 会将其包装成一个NSNumber
对象。
(2).若这几个方法都没有找到,则会调用+ (BOOL)accessInstanceVariablesDirectly
方法判断是否允许取成员变量的值。如果该方法返回NO,直接调用- (nullable id)valueForUndefinedKey:(NSString *)key
方法,默认是崩溃。
若返回YES,会按先后顺序取_key
、_isKey
、 key
、isKey
的值。
(3).返回YES时,_key
、_isKey
、key
、isKey
的值都没取到,调用- (nullable id)valueForUndefinedKey:(NSString *)key
方法。