【靠谱程序员#5】《招聘一个靠谱的程序员》个人解答

554 阅读7分钟

系列:

【靠谱程序员#0】《招聘一个靠谱的程序员》个人解答

【靠谱程序员#1】《招聘一个靠谱的程序员》个人解答

【靠谱程序员#2】《招聘一个靠谱的程序员》个人解答

【靠谱程序员#3】《招聘一个靠谱的程序员》个人解答

【靠谱程序员#4】《招聘一个靠谱的程序员》个人解答

【靠谱程序员#5】《招聘一个靠谱的程序员》个人解答

【靠谱程序员#6】《招聘一个靠谱的程序员》个人解答(end)

41. 如何用GCD同步若干个异步调用?(如根据若干个url异步加载多张图片,然后在都下载完成后合成一张整图)

答:

方法一:

使用Dispatch Group追加block到Global Group Queue,这些block如果全部执行完毕,就会执行Main Dispatch Queue中的结束处理的block。

dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);
dispatch_group_t group = dispatch_group_create();
dispatch_group_async(group, queue, ^{ /*加载图片1 */ });
dispatch_group_async(group, queue, ^{ /*加载图片2 */ });
dispatch_group_async(group, queue, ^{ /*加载图片3 */ }); 
dispatch_group_notify(group, dispatch_get_main_queue(), ^{
    // 合并图片
});

方法二:

dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);
dispatch_group_t group = dispatch_group_create();
dispatch_group_enter(group);
dispatch_async(queue, ^{
    /*加载图片1 */
    dispatch_group_leave(group);
});
dispatch_group_enter(group);
dispatch_async(queue, ^{
    /*加载图片2 */
    dispatch_group_leave(group);
});
dispatch_group_enter(group);
dispatch_async(queue, ^{
    /*加载图片3 */
    dispatch_group_leave(group);
});

dispatch_group_notify(group, dispatch_get_main_queue(), ^{
    // 合并图片
});

方法三:

/* 创建并发队列 */
dispatch_queue_t concurrentQueue = dispatch_queue_create("com.xby.queue", DISPATCH_QUEUE_CONCURRENT);
dispatch_async(concurrentQueue, ^(){
    /*加载图片1 */
});
dispatch_async(concurrentQueue, ^(){
    /*加载图片2 */
});
/* 添加barrier障碍操作,会等待前面的并发操作结束,并暂时阻塞后面的并发操作直到其完成 */
dispatch_barrier_async(concurrentQueue, ^(){
    NSLog(@"OperationBarrier!");
});

dispatch_async(dispatch_get_main_queue, ^(){
    // 合并图片
});

42. dispatch_barrier_async的作用是什么?

答:

在并行队列中,为了保持某些任务的顺序,需要等待一些任务完成后才能继续进行,使用 barrier 来等待之前任务完成,避免数据竞争等问题。 dispatch_barrier_async函数会等待追加到Concurrent Dispatch Queue并行队列中的操作全部执行完之后,然后再执行 dispatch_barrier_async函数追加的处理,等 dispatch_barrier_async 追加的处理执行结束之后,Concurrent Dispatch Queue才恢复之前的动作继续执行。

直白一点就是dispatch_barrier_async是一个栅栏,要等到前面所有操作都完成,这个栅栏才会开放,继续后面的操作。

(注意:使用 dispatch_barrier_async ,该函数只能搭配自定义并行队列 dispatch_queue_t 使用。不能使用: dispatch_get_global_queue ,否则 dispatch_barrier_async 的作用会和 dispatch_async 的作用一模一样。 )

43. 苹果为什么要废弃dispatch_get_current_queue?

答:

dispatch_get_current_queue容易造成死锁

详细说明

44. 以下代码运行结果如何?

- (void)viewDidLoad {
    [super viewDidLoad];
    NSLog(@"1");
    dispatch_sync(dispatch_get_main_queue(), ^{
        NSLog(@"2");
    });
    NSLog(@"3");
}

答:

只输出:1 ,发生主线程锁死。

同步加主队列等于死锁

45. addObserver:forKeyPath:options:context:各个参数的作用分别是什么,observer中需要实现哪个方法才能获得KVO回调?

// 添加键值观察
/*
1 观察者,负责处理监听事件的对象
2 观察的属性
3 观察的选项
4 上下文
*/
[self.person addObserver:self forKeyPath:@"name" options:NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld context:@"Person Name"];

observer中需要实现一下方法:

// 所有的 kvo 监听到事件,都会调用此方法
/*
 1. 观察的属性
 2. 观察的对象
 3. change 属性变化字典(新/旧)
 4. 上下文,与监听的时候传递的一致
 */
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context;

46. 如何手动触发一个value的KVO。

答:

手动在调用

// 设置前
[self willChangeValueForKey:@"keyName"]; 
// 设置后
[self didChangeValueForKey:@"keyName"]; 

拓展:

“手动触发”与“自动触发”

自动触发是指类似这种场景:在注册 KVO 之前设置一个初始值,注册之后,设置一个不一样的值,就可以触发了。

想知道如何手动触发,必须知道自动触发 KVO 的原理:

键值观察通知依赖于 NSObject 的两个方法: willChangeValueForKey:didChangevlueForKey:。 在一个被观察属性发生改变之前,willChangeValueForKey: 一定会被调用,这就 会记录旧的值。而当改变发生后, observeValueForKey:ofObject:change:context:会被调用,继而 didChangeValueForKey: 也会被调用。如果可以手动实现这些调用,就可以实现“手动触发”了。

那么“手动触发”的使用场景是什么?一般我们只在希望能控制“回调的调用时机”时才会这么做。

具体做法如下:

如果这个 value 是 表示时间的 self.now ,那么代码如下:最后两行代码缺一不可。

相关代码已放在仓库里。

//  .m文件
//  Created by https://github.com/ChenYilong
//  微博@iOS程序犭袁(http://weibo.com/luohanchenyilong/).
//  手动触发 value 的KVO,最后两行代码缺一不可。

//@property (nonatomic, strong) NSDate *now;
- (void)viewDidLoad {
   [super viewDidLoad];
   _now = [NSDate date];
   [self addObserver:self forKeyPath:@"now" options:NSKeyValueObservingOptionNew context:nil];
   NSLog(@"1");
   [self willChangeValueForKey:@"now"]; // “手动触发self.now的KVO”,必写。
   NSLog(@"2");
   [self didChangeValueForKey:@"now"]; // “手动触发self.now的KVO”,必写。
   NSLog(@"4");
}

但是平时我们一般不会这么干,我们都是等系统去“自动触发”。“自动触发”的实现原理:

比如调用 setNow: 时,系统还会以某种方式在中间插入 wilChangeValueForKey:didChangeValueForKey:observeValueForKeyPath:ofObject:change:context:的调用。 大家可能以为这是因为 setNow: 是合成方法,有时候我们也能看到有人这么写代码:

- (void)setNow:(NSDate *)aDate {
   [self willChangeValueForKey:@"now"]; // 没有必要
   _now = aDate;
   [self didChangeValueForKey:@"now"];// 没有必要
}

这完全没有必要,不要这么做,这样的话,KVO代码会被调用两次。KVO在调用存取方法之前总是调用 willChangeValueForKey:,之后总是调用 didChangeValueForkey:。怎么做到的呢?答案是通过 isa 混写(isa-swizzling)。下文《apple用什么方式实现对一个对象的KVO?》会有详述。

47. 若一个类有实例变量 NSString *_foo ,调用setValue:forKey:时,可以以foo还是 _foo 作为key?

答:

都可以。

设值:

设置值的时候会先查找set<名字>方法,如果没有,判定+ (BOOL)accessInstanceVariablesDirectly方法是不是返回YES,若为NO,会调用setValue:forUndefinedKey:方法,若为YES,则会依次判定_<名字>,_is<名字>,<名字>,is<名字>的成员变量,都不存在,同样会调用setValue:forUndefinedKey:方法。

读值:

先查找get<名字>,<名字>,is<名字>等getter方法,遇到基础数据类型就自动包装成对象类型;如果没找到这些getter方法,会调用代理集合NSKeyValueArray(NSArray的子类)的一些方法:countOf,objectInAtIndex或AtIndexes;如果代理集合的方法没有找到,会调用NSSet的方法:countOf,enumeratorOf,memberOf;若还是没找到,会按照set的方法判定+ (BOOL)accessInstanceVariablesDirectly,如果为NO,则调用valueForUndefinedKey:,如果返回YES(默认行为),那么和先前的设值一样,会按_<名字>,_is<名字>,<名字>,is<名字>的顺序搜索成员变量名,如果还是没有,同样调用调用valueForUndefinedKey:

48. KVC的keyPath中的集合运算符如何使用?

答:

  • 必须用在集合对象上或普通对象的集合属性上
  • 简单集合运算符有@avg, @count , @max , @min ,@sum,
  • 格式 @"@sum.age"或 @"集合属性.@max.age"

49. KVC和KVO的keyPath一定是属性么?

答:

KVC支持属性,实例变量(_<名字>),_is<名字>,is<名字>以及<名字> KVO默认只支持属性,但是可以让其手动支持实例变量

步骤:

a.设值实例变量

@interface Student : NSObject {
    NSString *_age;
}

b.手动实现setAge和age方法,在setAge方法中手动触发:

- (void)setAge:(NSString *)age {
    [self willChangeValueForKey:@"age"];    //手动调用
    _age = age;
    [self didChangeValueForKey:@"age"];     //手动调用
}

c.过滤这个实例变量:

+ (BOOL)automaticalNotifiesObserversForKey:(NSString *)key {
    if ([key isEqualToString:@"age"]) { //过滤
        return NO;
    }
    return [super automaticalNotifiesObserversForKey:key];
}

示例

50. 如何关闭默认的KVO的默认实现,并进入自定义的KVO实现?

答:

个人觉得可以参考49题,使用

+ (BOOL)automaticalNotifiesObserversForKey:(NSString *)key {
    return NO;  //全部过滤掉
}

全过滤掉,然后runtime方法替换- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context;,然后在替换的方法做一些自定义的事。

可行性未知???,还是看下面的参考链接吧)

请参考:

《如何自己动手实现 KVO》

KVO for manually implemented properties

下一篇:【靠谱程序员#6】《招聘一个靠谱的程序员》个人解答(end)