《招一个靠谱的iOS》46-50

本人参考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个参数的作用分别是:

  1. observer 观察者,负责处理监听事件的对象
  2. keyPath 被观察的属性,不能使nil
  3. options 设定通知观察者时传递的属性值,是传改变前的,还是传改变后的,具体有四种模式:NSKeyValueObservingOptionNew(change会接收到观察属性的新值)、NSKeyValueObservingOptionOld(change会接收到观察属性的旧值)、NSKeyValueObservingOptionInitial(回调方法会在观察属性初始化的时候调用,但不会接收到这个初始值,除非和NSKeyValueObservingOptionNew选项一起使用)、NSKeyValueObservingOptionPrior(该选项会触发两次,一次是观察属性改变前,一次是改变后,可以配合- (void)willChangeValueForKey:(NSString *)key一起使用)
  4. 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;
  1. keyPath 被观察的属性,其不能为nil
  2. object 被观察的对象,即哪个对象的属性被改了
  3. change 属性的修改状况,根据上面的方法中option参数,传不同的值
  4. 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中的集合运算符如何使用?

  1. 必须用在集合对象或普通对象的集合属性上;
  2. 简单集合运算符有@avg,@count,@max,@min,@sum;
  3. 格式@“@sum.age”或@“集合属性[email protected]”;

你可能感兴趣的:(《招一个靠谱的iOS》46-50)