美文网首页
RunLoop理解与应用

RunLoop理解与应用

作者: GY1994 | 来源:发表于2017-07-13 15:38 被阅读44次

什么是RunLoop

简单的来说就是运行循环、跑圈。

  • 保持着程序的持续运行并接受用户输入
  • 处理APP中的各种事件(触摸,定时器,Selector)
  • 节省了CPU资源,提高程序性能,即该做事时做事,该休息时休息。

RunLoop对象

iOS中有2套API来访问RunLoop

  • Foundation
    NSRunloop
  • Core Foundation
    CFRunLoopRef

CFRunLoopRef是用C语言写的更底层一些。NSRunLoop其实是对CFRunLoopRef的一个简单封装,CFRunLoopRef的性能高一点

获取RunLoop对象

  • NSRunLoop
[NSRunLoop currentRunLoop];获得当前线程的RunLoop对象
[NSRunLoop mainRunLoop];获得主线程的RunLoop对象
  • CFRunLoopRef
CFRunLoopGetCurrent();//获得当前线程的RunLoop对象
CFRunLoopGetMain();//获得主线程的RunLoop对象

RunLoop与线程间的关系

  • 每条线程都有唯一的一个与之对应的RunLoop对象
  • 主线程的RunLoop已经自动创建完成,子线程的需要手动创建
  • RunLoop在第一次获取时创建,在线程结束时销毁

CFRunLoopRef

image.png
  • CFRunLoopRefModeRef
    代表RunLoop的运行模式。
kCFRunLoopDefaultMode //app的默认Mode,通常主线程是在这个Mode下运行
UITrackingRunLoopMode //界面跟踪 Mode,用于 ScrollView 追踪触摸滑动,保证界面滑动时不受其他 Mode 影响
UIInitializationRunLoopMode //在刚启动 App 时第进入的第一个 Mode,启动完成后就不再使用
GSEventReceiveRunLoopMode //接受系统事件的内部 Mode,通常用不到
kCFRunLoopCommonModes //这是一个占位用的Mode,不是一种真正的Mode
  • CFRunLoopObserverRef
    观察者,能够监听RunLoop的状态变化。
kCFRunLoopEntry //即将进入loop
kCFRunLoopBeforeTimers //即将处理timer
kCFRunLoopBeforeSources //即将处理source
kCFRunLoopBeforeWaiting //即将进入休眠
kCFRunLoopAfterWaiting //即将休眠中唤醒
kCFRunLoopExit  //即将退出loop
kCFRunLoopAllActivities //所有状态
  • CFRunLoopSourceRef
    RunLoop要处理的事件源,定义了两种Source
    Source0:处理APP内部时间,APP自己负责管理(触发),如UIevent...基于非Port的
    Source1:由RunLoop和内核管理的系统的一些事件,基于Port的。
  • CFRunLoopTimerRef
    是基于事件的触发器,即NSTimer

创建一个RunLoop之后,有默认的运行模式mode,也可以为RunLoop指定运行模式,RunLoop启动必须得有运行模式,而且在运行模式中必须还有timer或是source事件其中之一,否则RunLoop就会退出。CFRunLoopModeRef代表RunLoop的运行模式
一个 RunLoop 包含若干个Mode,每个Mode又包含若干个SourceTimerObserver
每次RunLoop启动时,只能指定其中一个Mode,这个Mode被称作 CurrentMode
如果需要切换Mode,只能退出RunLoop,再重新指定一个Mode进入,这样做主要是为了分隔开不同组的SourceTimerObserver,让其互不影响。

image.png image.png
     /*
        1参:怎么分配存储空间
        2参:眼监听的状态 kCFRunLoopAllActivities 所有状态
        3参:是否持续监听
        4参:优先级 总是传0
        5参:当状态改变时候的回调
     */
    CFRunLoopObserverRef observer = CFRunLoopObserverCreateWithHandler(CFAllocatorGetDefault(), kCFRunLoopAllActivities, YES, 0, ^(CFRunLoopObserverRef observer, CFRunLoopActivity activity) {
        switch (activity) {
            case kCFRunLoopEntry:
                NSLog(@"即将进入runloop");
                break;
            case kCFRunLoopBeforeTimers:
                NSLog(@"即将处理timer事件");
                break;
            case kCFRunLoopBeforeSources:
                NSLog(@"即将处理source事件");
                break;
            case kCFRunLoopBeforeWaiting:
                NSLog(@"即将进入睡眠");
                break;
            case kCFRunLoopAfterWaiting:
                NSLog(@"被唤醒");
            case kCFRunLoopExit:
                NSLog(@"runloop退出");
                break;
            default:
                break;
        }
    });
    /*
        1参:要监听那个runloop
        2参:观察者
        3参:运行模式
     */
    CFRunLoopAddObserver(CFRunLoopGetCurrent(), observer, kCFRunLoopDefaultMode);
    CFRelease(observer);//移除

NSRunLoop

  • NSTimer问题
    NSTimer使用scheduledTimerWithTimeInterval此方法创建的定时器默认加到了NSRunLoop中,并且设置运行模式为默认default,但是如果有滚动事件的时候,定时器就会停止工作。
    解决办法:更改NSRunLoop的Mode为NSRunLoopCommonModes,滚动拖拽页面时定时器仍然工作。
//1.创建定时器
NSTimer *timer =  [NSTimer timerWithTimeInterval:2.0 target:self selector:@selector(run) userInfo:nil repeats:YES];
NSRunLoop *currentRunloop = [NSRunLoop currentRunLoop];
//添加至RunLoop并更改运行模式
[[NSRunLoop currentRunLoop] addTimer:timer forMode:NSRunLoopCommonModes];
//开启runloop
[currentRunloop run];
  • 线程常驻
    创建类GYThread继承NSThreadGYThread.m中添加dealloc
 - (void)dealloc{
    NSLog(@"线程被销毁了");
 } 

初始化线程,并创建该线程的RunLoop,s

_thread = [[GYThread alloc]initWithTarget:self selector:@selector(threadtest) object:nil];
 _thread.name = @"geyang";
[_thread start];

 - (void)threadtest{
    NSLog(@"--%@--", [NSThread currentThread]);
    //启动该�线程的runloop
    [[NSRunLoop currentRunLoop]addPort:[NSPort port] forMode:NSDefaultRunLoopMode];
    [[NSRunLoop currentRunLoop]run];
 }
//检测线程是否被销毁
 - (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{
    [self performSelector:@selector(threadAgain) onThread:_thread withObject:nil waitUntilDone:NO];
}
 - (void)threadAgain{
    NSLog(@"---%@---", [NSThread currentThread]);
}

此时,我们可以点击屏幕可看到打印结果

2017-07-13 15:23:24.382 RunLoop与线程[9290:1054512] ---<GYThread: 0x608000269300>{number = 3, name = geyang}---

说明该线程并没有被销毁

相关文章

  • iOS开发经验(18)-Runloop

    目录 Runloop RunLoop 与线程 个人理解总结 应用场景 1. 什么是RunLoop 基本作用 保持程...

  • RunLoop理解与应用

    什么是RunLoop 简单的来说就是运行循环、跑圈。 保持着程序的持续运行并接受用户输入 处理APP中的各种事件(...

  • runloop

    1.RunLoop的应用Runloop应用 2.深入理解RunLoop 2.动态计算UITableViewCell...

  • NSRunLoop

    【iOS程序启动与运转】- RunLoop个人小结 RunLoop总结:RunLoop的应用场景(三) 走进Run...

  • RunLoop学习笔记

    参考深入理解RunLoop深入研究 Runloop 与线程保活RunLoop分享by孙源 RunLoop的概念 R...

  • Runloop的应用与深入理解

    RunLoop的概念 一般来讲,一个线程一次只能执行一个任务,执行完成后线程就会退出。如果我们需要一个机制,让线程...

  • Runloop分析

    RunLoop 参考:深入理解RunLoop ibireme:《深入理解RunLoop》 Runloop 的概念 ...

  • iOS 常见面试题--runloop

    一、什么是runloop 二、runloop的基本作用 三、应用范畴 四、runloop与线程之间的关系 五、Ru...

  • Runloop实际应用中如何优化app流畅度

    这次主要讲的Runloop的实际应用,基础的内容就不在这介绍了,详细的文章可以查看深入理解RunLoop RunL...

  • Runloop学习资料整理

    基于runloop的线程保活、销毁与通信 深入理解RunLoop developer.apple.com/libr...

网友评论

      本文标题:RunLoop理解与应用

      本文链接:https://www.haomeiwen.com/subject/snywhxtx.html