这次给大家分享一些面试题,你知道答案吗?
第一道面试题:
请看下面的代码,请问输出的结果是什么?
第二道面试题:
请看下面的代码,请问输出的结果是什么?
第三道面试题:
请看下面的代码,请问输出的结果是什么?
第四道面试题:
请看下面的代码,请问着2块代码会发生什么事?
面试题答案
1.第一道:
相信我们心中都是已经有答案了,现在我们一个一个来看看每一道面试题,其实考察的还是我们的基础的问题,涉及的都是某一个知识点,首先第一道.这样我们先看一下运行结果:1和3
结果是1和3,而test2方法并没有执行,这时候我换一种写法,如下:我们再看一下结果.
afterDelay去掉以后立刻就会输出1,2,3.首先我们要知道[self performSelector:@selector(test2) withObject:nil];这里的执行本质就是objc_msgSend(self,@selector(test2))就是这种,再说白点就是[self test2];(这里可以查看objc源码的实现,直接查找performSelector的方法实现就能找到).
而[self performSelector:@selector(test2) withObject:nil afterDelay:.0];就有点不一样的,大家可以把上面的异步线程的里面的代码全部拿到主线程来操作,你会发现都是有输出的.会输出1,2,3(顺序可能不一样,但都会输出).这时候你可以点击进去看里面的定义,你会发现它是属于RunLoop里面的模块,而且在objc里面你是找不到源码的.这时候我们就可以参照上个博客说的GNUstep的源码(GNUstep上个博客有介绍),我们看源码如下:
所以很明显,它的底层是用了NSTimer定时器.比如时间我写2.0就是2秒以后做事情.而我们知道定时器是要加在Runloop里面.这下就很清楚了,为什么主线程可以,因为我们知道主线程的Runloop是默认开启的,子线程是默认没有Runloop的,所以会出现这种情况.定时器没法工作,它那段代码是往runloop中添加了一个定时器任务.如果想让它工作,我们只要添加runloop即可.
2.第二道:
我们先看一下第二道面试题的输出结果
输出1以后,执行performSelector任务的时候,直接崩溃了,因为我们知道,调用start,它就会执行block任务,执行完以后,线程的任务就算完成了,线程自动会退出,这时候你去让一个已经退出的线程安排任务,肯定会出问题.
3.第三道:
第三道也可以说是第二道的升华,因为有些人可能会想,既然thread会退出,那我就用一个强指针指着,不让它退出,不就行了吗?真的可以吗?我们试试.
你会发现这样写还是一样,还是会崩溃,这是为什么呢?
这是因为,强指针只是让它不销毁,让这个线程对象还是在内存中,但是它不能做事情了,它已经没有用了,而只有添加RunLoop才是保证线程处于激活的状态,保证它的生命周期不结束,还能继续在这个线程中执行线程任务.所以必须使用RunLoop才能保住线程的生命周期.
而第三道面试题,我是加了runloop,所以就能正常执行,而第三道面试题,第一块是只执行了1次performSelector,它是不会崩溃正常执行的.这样,我还是截图代码运行结果吧,请看下面:
为什么会有上面的结果呢?原因它还是关于Runloop的一个线程保活的问题(也就是控制线程的生命周期),这里就不细说了,之前的博客都有介绍,有兴趣的可以看一下.你了解RunLoop线程保活吗?已封装好,2句代码直接使用里面详细介绍run和runMode区别,知道这个就能很清楚解答这道面试题
4.第四道:
这个是涉及tagged pointer的知识点,如果你了解,这道面试题就相对来说简单些,这样我们先看运行结果:
我们可以看到上面的2张图,第一张直接崩溃了,第二张图的运行结果没有什么问题,要解答这个面试题,你要知道tagged pointer这个东西:(这里大致说一下,因为我后面的博客会详细介绍这个tagged pointer)
tagged pointer: 从 64bit 开始,苹果引入了 tagged pointer 计数,用于优化 NSNumber , NSDate , NSString 等小对象的存储,没有这个数据之前,NSNumber 等对象需要动态分配内存,维护引用计数,NSNumber 指针存储的是堆中NSNumber对象的地址值,而引入了这个计数之后,NSNumber 指针里面存储的数据是 : tag + data ,也就是直接将数据存储在指针中。这样做特别节省空间。如果这个数据特别大,指针存储不下这个数,那么会恢复之前的方式,存储在堆区,然后指针存放堆区的地址。
我们知道存在堆区的话,给TestStr赋值内部会类似下面的访问:
-(void)setTestStr:(NSString *)TestStr{
if (_TestStr != TestStr) {
[_TestStr release];
_TestStr = [TestStr copy];
}
}
这样的话就可能有多个线程同时去 [_TestStr release],如果_TestStr已经释放了,又去访问就会直接崩溃.为了避免这个问题,我们可以使用线程同步技术,我上个博客写了好多种技术,我们随便用一个即可解决.
而self.TestStr = [NSString stringWithFormat:@"abc"]这个就是使用了tagged pointer技术,所以不会涉及到释放,就不会出现崩溃的问题.(tagged pointer后面的博客我会详细介绍)