Objective-C进阶
2020-06-01 本文已影响0人
again_onceagain
分类
Q1: 你用分类都做了那些事?
- 声明私有方法
- 分解体积庞大的类文件
- 把Framework的私有方法公开化
Q2: 特点
- 运行时决议
- 可以为系统类添加分类
Q3: 分类中可以添加哪些内容?
- 实例方法
- 类方法
- 协议
- 属性(只定义了getter/setter方法)
分类数据结构:
image.png- 分类添加的方法可以“覆盖”原类方法
- 同名分类方法谁能生效取决于编译顺序(最后编译的类最先生效)
- 名字相同的分类会引起编译报错
关联对象
Q1: 能否给分类添加“成员变量”?
id objc_getAssociatedObject(id object, const void * key)
void objc_setAssociatedObject(id object, const void * key, id value, objc_AssociationPolicy policy)
voi objc_removeAssociatedObjects(id object)
Q2: 成员变量添加到哪呢?
关联对象的本质:
关联对象由AssociationsManager
管理并在AssociationHashMap
存储。
所有对象的关联内容都在同一个全局容器
中。
扩展(Extension)
Q1: 一般用扩展做什么?
- 声明私有属性
- 声明私有方法
- 声明私有成员变量
Q2:特点
- 编译时决议
- 只以声明的形式存在,多数情况下寄生于宿主类的.m中。
- 不能为系统类添加扩展。
代理(Delegate)
-
准确的说是一种软件设计模式
-
ios当中以
@protocol
形式提现 -
传递方式
image.png一对一
-
一般声明以
image.pngweak
以规避循环引用
通知(NSNotification)
- 是使用
观察者模式
来实现的用于跨层传递消息的机制。 - 传递方式为
一对多
。
image.png
Q1: 如何实现通知机制?
image.png
KVO
- KVO是Key-value observing的缩写。
- KVO是Objective-C对
观察者设计模式
的又一实现。 - Apple使用了isa混写(
isa-swizzling
)来实现KVO。
- (void)addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(nullable void *)context;
- (void)removeObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath context:(nullable void *)context API_AVAILABLE(macos(10.7), ios(5.0), watchos(2.0), tvos(9.0));
- (void)removeObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath;
image.png
重写的setter添加的方法
- (void)willChangeValueForKey:(NSString *)key;
- (void)didChangeValueForKey:(NSString *)key;
- (void)setValue:(int)value {
//直接为成员变量赋值
[self willChangeValueForKey:@"value"];
_value += 1;
[self didChangeValueForKey:@"value"];
}
Q1: 通过KVC设置value, KVO能否生效?
YES.(KVC实现原理)
Q2: 通过成员变量直接赋值value, KVO能否生效?
NO. 但是可以手动触发KVO, 代码如下
- (void)increase {
//直接为成员变量赋值
[self willChangeValueForKey:@"value"];
_value += 1;
[self didChangeValueForKey:@"value"];
}
总结
- 使用setter方法改变值KVO才会生效。
- 使用
setValue:forKey:
改变值KVO才会生效。 - 成员变量直接修改需
手动添加
KVO才会生效。
KVC
KVC是Key-value coding的缩写
- (nullable id)valueForKey:(NSString *)key;
- (void)setValue:(nullable id)value forKey:(NSString *)key;
KVC破坏了面向对象的编程思想(封装性)
实现流程
Accessor Method
- <getKey>
- <key>
- <isKey>
Instance var
- _key
- _isKey
- key
- isKey
属性关键字
- 读写权限
- readonly
readwrite
- 原子性
atomic
- nonatomic
- 引用计数
- retain/
strong
-
assign
/unsafe_unretained - weak
- copy
- retain/
assgin
- 修饰基本数据类型, 如int, BOOL等。
- 修饰对象类型时,不改变其引用计数。
- 会产生悬悬垂指针。
weak
- 不改变被修饰对象的引用计数。
- 所知对象在呗释放之后会自动置为nil。
copy
-
浅拷贝
image.png
浅拷贝就是对内存地址的复制,让目标对象指针和源对象指向同一片内存空间。
- 深拷贝
image.png
深拷贝让目标对象指针和源对象指针指向两片内容相同
的内存空间。
如何区分深拷贝vs浅拷贝
- 是否开辟了新的内存空间
- 是否影响了引用计数
- 可变对象的copy和mutableCopy都是深拷贝。
- 不可变对象的copy是浅拷贝, mutableCopy是深拷贝。
- copy方法返回的都是不可变对象。
- mutableCopy方法返回的都是可变对象。
@property(copy) NSMutableArray *array?
- 如果赋值过来的是NSMutableArray, copy之后还是NSArray
- 如果赋值过来的NSArray, copy之后是NSArray
MRC下如何重写retain修饰变量的Setter方法?
if (_obj != obj) {
[_obj release];
_obj = [obj retain];
}
请简述分类的实现原理。
KVO的实现原理是怎么样的?
能否为分类添加成员变量?