系列:
【靠谱程序员#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;,然后在替换的方法做一些自定义的事。
(可行性未知???,还是看下面的参考链接吧)