iOS基础面试题之RunLoop篇

126 阅读4分钟

RunLoop是什么,有什么作用,如何获取?

  • 定义
    • RunLoop的实质是一个死循环,用于保证程序的持续运行,只有当程序退出的时候才会结束(由main函数开启主线程的RunLoop)
  • 作用
    • 保持程序的持续运行
    • 处理App中的各种事件(触摸、定时器、Selector事件)
    • 节省CPU资源,提高程序性能(该做事做事,没事做休息)
  • 获取方法
    • 使用NSRunLoop(面向对象)或者CFRunLoopRef(底层C语言)

RunLoop的原理

  • RunLoop开启一个循环事件,并接受输入事件,接受的事件来自两种不同的来源:
    • 输入源(input source)(传递异步事件)
    • 定时源(timer source)(传递同步事件)
  • RunLoop接收到消息后采用handlePort、customSrc、mySelector和timerFired等四个方法处理对应的事件
  • 当RunLoop没有接收到消息时,则进入休眠状态,以保持程序持续运行

本人5年iOS开发经验,曾就职于阿里巴巴。 善于把艰涩的iOS知识转化为通俗易懂的白话文字,同时也欢迎大家加入小编的iOS交流群 834688868 ,群里会提供相关面试资料,书籍欢迎大家入驻!

image

RunLoop接收几种输入源,系统默认定义了几种模式?

  • 输入源有两种
    • 基于端口的输入源(port)
    • 自定义的输入源(custom)
  • 系统定义的RunLoop模式有五种,最常用的有三种,如下所示:
    • NSDefaultRunLoopMode
      • 默认模式,主线程中默认是NSDefaultRunLoopMode
    • UITrackingRunLoopMode
      • 视图滚动模式,RunLoop会处于该模式下
    • NSRunLoopCommonModes
      • 并不是真正意义上的Mode,是一个占位用的“Mode”,默认包含了NSDefaultRunLoopMode和UITrackingRunLoopMode两种模式

RunLoop模式的原理和使用注意点?

  • 原理和注意点
    • 一个RunLoop包含若干个Mode,每个Mode又包含若干个Source、Observer、Timer(如下图所示)
    • 每次RunLoop启动,只能指定一个Mode,这个Mode被称为CurrentMode
    • 如果需要切换Mode,只能退出Loop,再重新指定一个Mode进入, 以使不同组之间的Source、Observer、Timer互不受影响

image

RunLoop和线程有什么关系

  • RunLoop与线程是一一对应的
  • 程序启动时,主线程默认会自己创建RunLoop,并设置为Default模式
  • 创建子线程时,必须获取当前线程的RunLoop并启动它
NSRunLoop *loop = [NSRunLoop currentRunLoop];
[loop run];


NSTimer和RunLoop的关系?

  • NSTimer需要添加到Runloop中, 才能执行的情况
  NSTimer *timer = [NSTimer timerWithTimeInterval:1.f target:self selector:@selector(update) userInfo:nil repeats:YES];
    [[NSRunLoop currentRunLoop] addTimer:timer forMode:NSDefaultRunLoopMode];

  • NSTimer默认被添加到Runloop中, 直接执行的情况
[NSTimer scheduledTimerWithTimeInterval:1.f target:self selector:@selector(update) userInfo:nil repeats:YES];

NSTimer准确吗,如果不准确,如何设计一个准确的timer?

  • 不准确
  • 准确的Timer应该和当前线程的RunLoopMode保持一致

TableView/ScrollView/CollectionView滚动时为什么NSTimer会停止?

  • 一个RunLoop不能同时共存两个mode
  • 当滚动视图滚动时,当前RunLoop处于UITrackingRunLoopMode,
  • NSTimer的RunLoopMode和当前线程的RunLoopMode不一致,所以会停止
  • 解决方法:将timer的runloopMode改为UITrackingRunLoopMode或者NSRunLoopCommonModes

如果NSTimer在分线程中创建,会发生什么,应该注意什么?

  • NSTimer没有启动
    • 在主线程中,系统默认创建并启动主线程的runloop
    • 在分线程中,系统不会自动启动runloop,需要手动启动
  • 解决方法:
    • 启动分线程的runLoop

在异步线程中下载很多图片,如果失败了,该如何处理?请结合RunLoop来谈谈解决方案

  • 在异步线程中启动一个RunLoop重新发送网络请求,下载图片

如果程序启动就需要执行一个耗时操作,你会怎么做?

  • 开启一个异步的子线程,并启动它的RunLoop来执行该耗时操作

runloop与autoreleasepool的关系

如果在分线程中启动一个异步请求,会有什么问题?

判断其是否请求结束,如果未结束,要保持当前线程一直启动,直到结束

while(!isFinish)
     {
       [[NSRunLoop currentRunLoop] runMode:NSDefaultRunLoopMode beforeDate:[NSDate distantFuture]];
     }

程序启动时,runloop是如何工作的?如果程序启动就需要执行一个耗时操作,你会怎么做?

程序启动时,系统默认创建并启动主线程的runloop,runloop会默认创建两个Observe来进行监听runloop的进出和睡眠,有事情的时候就去做,没事的休眠

(线程(创建)-->runloop将进入-->最高优先级OB创建释放池-->runloop将睡-->最低优先级OB销毁旧池创建新池-->runloop将退出-->最低优先级OB销毁新池-->线程(销毁))

线程刚创建时并没有runloop,如果你不主动去获取,那么一直都不会有。

耗时操作可以放在分线程中进行,结束后回到主线程

作者:Stars木木 链接:www.jianshu.com/p/848ea3817…