耶,这些面试题有点歪,你会吗?

1,976 阅读6分钟

这次给大家分享一些面试题,你知道答案吗?

第一道面试题:

请看下面的代码,请问输出的结果是什么?

第二道面试题:

请看下面的代码,请问输出的结果是什么?

第三道面试题:

请看下面的代码,请问输出的结果是什么?

第四道面试题:

请看下面的代码,请问着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后面的博客我会详细介绍)

接下来我会继续努力编写其他博客,您的支持就是我最大的动力!

如果觉得我写得对您有所帮助,请点赞关注我,我会持续更新😄

感谢支持🙏🙏🙏!

我是GDCoder,我们下期见!