本人参考GitHub《招聘一个靠谱的iOS》面试题参考答案(下)
46. 以下代码运行结果如何?
- (void)viewDidLoad
{
[super viewDidLoad];
NSLog(@"1");
dispatch_sync(dispatch_get_main_queue(), ^{
NSLog(@"2");
});
NSLog(@"3");
}
47. addObserver:forKeyPath:options:context:各个参数的作用分别是什么,observer中需要实现哪个方法才能获得KVO回调?
48. 如何手动触发一个value的KVO?
49. 若一个类有实例变量NSString *_foo
,调用setValue:forKey:时,可以以foo还是_foo
作为key?
50. KVC的keyPath中的集合运算符如何使用?
46. 以下代码运行结果如何?
- (void)viewDidLoad
{
[super viewDidLoad];
NSLog(@"1");
dispatch_sync(dispatch_get_main_queue(), ^{
NSLog(@"2");
});
NSLog(@"3");
}
只输出1,代码会产生死锁。
47. - (void)addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(nullable void *)context;
各个参数的作用分别是什么?observer中需要实现哪个方法才能获取KVO回调?
4个参数的作用分别是:
- observer 观察者,负责处理监听事件的对象
- keyPath 被观察的属性,不能使nil
- options 设定通知观察者时传递的属性值,是传改变前的,还是传改变后的,具体有四种模式:NSKeyValueObservingOptionNew(change会接收到观察属性的新值)、NSKeyValueObservingOptionOld(change会接收到观察属性的旧值)、NSKeyValueObservingOptionInitial(回调方法会在观察属性初始化的时候调用,但不会接收到这个初始值,除非和NSKeyValueObservingOptionNew选项一起使用)、NSKeyValueObservingOptionPrior(该选项会触发两次,一次是观察属性改变前,一次是改变后,可以配合
- (void)willChangeValueForKey:(NSString *)key
一起使用) - context 一些其他的需要传递给观察者的上下文信息,主要用于区分父类是否也监听了同一个属性,通常设置为nil。
// 添加键值观察
[self.person addObserver:self forKeyPath:@"name" options:NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld context:@"Person name"];
observer中需要实现下面的方法:
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context;
- keyPath 被观察的属性,其不能为nil
- object 被观察的对象,即哪个对象的属性被改了
- change 属性的修改状况,根据上面的方法中option参数,传不同的值
- context 上文传递的context属性的内容
48. 如何手动出发一个value的KVO?
“手动触发”与“自动触发”相对。
“自动触发”是指在KVO注册之前设置一个值,在注册之后,设置一个不一样的值,就可以触发KVO了。
想知道如何手动触发KVO,就必须知道自动触发KVO的原理:键值观察通知依赖于NSObject的两个方法:- (void)willChangeValueForKey:(NSString *)key
和- (void)didChangeValueForKey:(NSString *)key
。在一个被观察属性发生改变之前,- (void)willChangeValueForKey:(NSString *)key
一定会被调用,这就会记录旧的值。而当改变发生之后,- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context
方法会被调用,继而- (void)didChangeValueForKey:(NSString *)key
也会被调用。如果可以手动实现这些调用,就可以实现“手动触发”了。
手动触发的场景一般是我们在希望能控制“回调的调用时机”时才这么做。
具体方法如下,以value是表示时间的self.now为例,具体代码如下:
- (void)viewDidLoad {
[super viewDidLoad];
_now = [NSDate date];
[self addObserver:self forKeyPath:@"now" options:NSKeyValueObservingOptionNew context:nil];
[self willChangeValueForKey:@"now"]; // “手动触发”self.now的KVO必写
[self didChangeValueForKey:@"now"]; // “手动触发”self.now的KVO必写
}
但是一般我们不会这么干,我们都是等待系统去“自动触发”。
自动触发的实现原理:
以调用setNow:为例,系统还会以某种方式在中间插入- (void)willChangeValueForKey:(NSString *)key
、- (void)didChangeValueForKey:(NSString *)key
和- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context;
的调用。
49. 若一个类有实力变量NSString *_foo
,调用setValue:forKey:时,可以用foo还是_foo
作为key?
都可以。
50. KVC的keyPath中的集合运算符如何使用?
- 必须用在集合对象或普通对象的集合属性上;
- 简单集合运算符有@avg,@count,@max,@min,@sum;
- 格式@“@sum.age”或@“集合属性.@max.age”;