iOS 面试题总结

1. app如何接收到触摸事件的

https://www.jianshu.com/p/f73b4bdc73c7

2. 为什么只有主线程的runloop是开启的

https://developer.aliyun.com/ask/282845

mian函数中调用UIApplicationMain,这里会创建一个主线程,用于UI处理,为了让程序可以一直运行,所以在主线程中开启一个runloop,让主线程常驻。

3. 为什么只在主线程刷新UI

https://www.cnblogs.com/ederwin/articles/10683385.html

从UIKit线程不安全说起

在UIKit中,很多类中大部分的属性都被修饰为nonatomic,这意味着它们不能在多线程的环境下工作,而对于UIKit这样一个庞大的框架,将其所有属性都设计为线程安全是不现实的,这可不仅仅是简单的将nonatomic改成atomic或者是加锁解锁的操作,还涉及到很多的方面:

假设能够异步设置view的属性,那我们究竟是希望这些改动能够同时生效,还是按照各自runloop的进度去改变这个view的属性呢?

假设UITableView在其他线程去移除了一个cell,而在另一个线程却对这个cell所在的index进行一些操作,这时候可能就会引发crash。

如果在后台线程移除了一个view,这个时候runloop周期还没有完结,用户在主线程点击了这个“将要”消失的view,那么究竟该不该响应事件?在哪条线程进行响应?

仔细思考,似乎能够多线程处理UI并没有给我们开发带来更多的便利,假如你代入了这些情景进行思考,你很容易得出一个结论:我在一个串行队列对这些事件进行处理就可以了。苹果也是这样想的,所以UIKit的所有操作都要放到主线程串行执行。

4. PerformSelectorrunloop的关系

https://www.jianshu.com/p/ab0c2e00435f

看文档中对这个API的注释是说,这个方法调用后,在当前runloop里设置了一个timer,来触发这个方法执行。而当前这个方法是在子线程中调用的,在子线程中runloop不是自动创建并跑起来的,需要手动调用,才会创建。因为这个在子线程中的调用没有创建runloop,所以就没有执行testPerform。

5. 如何使线程保活

https://www.jianshu.com/p/3ee42ff6e5ed

其实就是开启子线程的runloop

6. 实现原理

https://www.jianshu.com/p/badf5cac0130

KVO是通过isa-swizzling技术实现的(这句话是整个KVO实现的重点)。在运行时根据原类创建一个中间类,这个中间类是原类的子类,并动态修改当前对象的isa指向中间类。并且将class方法重写,返回原类的Class。所以苹果建议在开发中不应该依赖isa指针,而是通过class实例方法来获取对象类型。

KVO是基于runtime机制实现的

当某个类的属性对象第一次被观察时,系统就会在运行期动态地创建该类的一个派生类(该类的子类),在这个派生类中重写基类中任何被观察属性的setter 方法。派生类在被重写的setter方法内实现真正的通知机制

如果原类为Person,那么生成的派生类名为NSKVONotifying_Person

每个类对象中都有一个isa指针指向当前类,当一个类对象的第一次被观察,那么系统会偷偷将isa指针指向动态生成的派生类,从而在给被监控属性赋值时执行的是派生类的setter方法

键值观察通知依赖于NSObject 的两个方法: willChangeValueForKey: 和 didChangevlueForKey:;在一个被观察属性发生改变之前, willChangeValueForKey: 一定会被调用,这就 会记录旧的值。而当改变发生后,didChangeValueForKey: 会被调用,继而 observeValueForKey:ofObject:change:context: 也会被调用。

补充:KVO的这套实现机制中苹果还偷偷重写了class方法,让我们误认为还是使用的当前类,从而达到隐藏生成的派生类 

7. 如何手动关闭kvo

https://www.cnblogs.com/huangzs/p/7515653.html

+(BOOL)automaticallyNotifiesObserversForKey:(NSString *)key{

    if ([key isEqualToString:@"name"]) {

        return NO;

    }else{

        return [super automaticallyNotifiesObserversForKey:key];

    }

}

-(void)setName:(NSString *)name{


    if (_name!=name) {


        [self willChangeValueForKey:@"name"];

        _name=name;

        [self didChangeValueForKey:@"name"];

    }


}

8. 通过KVC修改属性会触发KVO

KVO是键值观察,当所监听的对象的属性发生改变时,会调用代理方法,知道这个属性是怎么变化的。

KVC是键值编码,对对象的属性就行取值赋值,

KVC能触发KVO,KVC在赋值的时候会会进行方法的调用,首先调用(_setKey,setKey方法),如果没有set方法,责直接内部自己调用设置属性(key,_key),内部就会监听到值的改变。

你可能感兴趣的:(iOS 面试题总结)