iOS

iOS-底层原理(19)-RunLoop详解-类,运行逻辑

2018-09-11  本文已影响61人  路飞_Luck
序言
什么是RunLoop

顾名思义

runloop.png
应用范畴

代码例子佐证

int main(int argc, const char * argv[]) {
    @autoreleasepool {
        // insert code here...
        NSLog(@"Hello, World!");
    }
    return 0;
}

执行完NSlog打印后,会即将退出程序

int main(int argc, char * argv[]) {
    @autoreleasepool {
        return UIApplicationMain(argc, argv, nil, NSStringFromClass([AppDelegate class]));
    }
}

等价于下面的伪代码

int main(int argc, char * argv[]) {
    @autoreleasepool {
        int retVal = 0;
        do {
            // 睡眠中等待消息
            int message = sleep_and_wait();
            // 处理消息
            retVal = process_message(message);
        } while (0 == retVal);
        
        return 0;
    }
}

程序并不会马上退出,而是保持运行状态

RunLoop的基本作用
二 RunLoop对象

iOS中有2套API来访问和使用RunLoop

NSRunLoopCFRunLoopRef都代表着RunLoop对

三 RunLoop与线程
四 获取RunLoop对象
[NSRunLoop currentRunLoop];  // 获得当前线程的RunLoop对象
[NSRunLoop mainRunLoop];  // 获得主线程的RunLoop对象
CFRunLoopGetCurrent();  // 获得当前线程的RunLoop对象
CFRunLoopGetMain();  // 获得主线程的RunLoop对象
五 RunLoop相关的类

Core Foundation中关于RunLoop的5个类

下图是这几种类的关系

image.png
5.1 CFRunLoopModeRef
5.2 CFRunLoopModeRef

常见的2种Mode

NSDefaultRunLoopModeUITrackingRunLoopMode才是真正存在的模式
NSRunLoopCommonModes并不是一个真的模式,它只是一个标记

通过打印当前主线程的runloop可以查看所有信息

NSLog(@"%@",[NSRunLoop mainRunLoop]);
六 RunLoop的运行逻辑
image.png

代码例子佐证如下

- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event {
    // 函数调用栈
    NSLog(@"111111");
}

打印结果


source0.png
// UI刷新
self.view.backgroundColor = [UIColor redColor];

UI刷新也是runloop即将进入休眠时才去刷新

七 RunLoop的运行逻辑
CFRunLoopObserverRef .png
01、通知Observers:进入Loop
02、通知Observers:即将处理Timers
03、通知Observers:即将处理Sources
04、处理Blocks
05、处理Source0(可能会再次处理Blocks)
06、如果存在Source1,就跳转到第8步
07、通知Observers:开始休眠(等待消息唤醒)
08、通知Observers:结束休眠(被某个消息唤醒)
  01> 处理Timer
  02> 处理GCD Async To Main Queue
  03> 处理Source1
09、处理Blocks
10、根据前面的执行结果,决定如何操作
  01> 回到第02步
  02> 退出Loop
11、通知Observers:退出Loop
RunLoop的运行逻辑 .png
7.1 添加Observer监听RunLoop的所有状态

代码如下

// 监听runloop状态的变化
void observeRunLoopActicities(CFRunLoopObserverRef observer, CFRunLoopActivity activity, void *info) {
    switch (activity) {
        case kCFRunLoopEntry:
            NSLog(@"kCFRunLoopEntry");
            break;
        case kCFRunLoopBeforeTimers:
            NSLog(@"kCFRunLoopBeforeTimers");
            break;
        case kCFRunLoopBeforeSources:
            NSLog(@"kCFRunLoopBeforeSources");
            break;
        case kCFRunLoopBeforeWaiting:
            NSLog(@"kCFRunLoopBeforeWaiting");
            break;
        case kCFRunLoopAfterWaiting:
            NSLog(@"kCFRunLoopAfterWaiting");
            break;
        case kCFRunLoopExit:
            NSLog(@"kCFRunLoopExit");
            break;
        default:
            break;
    }
}

// 监听runloop
- (void)observeRunloop {
    // kCFRunLoopCommonModes默认包括kCFRunLoopDefaultMode、UITrackingRunLoopMode
    // 创建Observer
    CFRunLoopObserverRef observer = CFRunLoopObserverCreate(kCFAllocatorDefault, kCFRunLoopAllActivities, YES, 0, observeRunLoopActicities, NULL);
    // 添加Observer到RunLoop中
    CFRunLoopAddObserver(CFRunLoopGetMain(), observer, kCFRunLoopCommonModes);
    // 释放
    CFRelease(observer);
}

// 验证触摸事件为source0
- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event {
    NSLog(@"开始触摸操作");
    dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2.0 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
        NSLog(@"开始定时器操作");
    });
}

运行结果如下

image.png
7.2 监听runloop的运行

拖拽一个textView至界面中

- (void)observeRunloopModeChagne {
    // 创建Observer
    CFRunLoopObserverRef observer = CFRunLoopObserverCreateWithHandler(kCFAllocatorDefault, kCFRunLoopAllActivities, YES, 0, ^(CFRunLoopObserverRef observer, CFRunLoopActivity activity) {
        switch (activity) {
            case kCFRunLoopEntry: {
                CFRunLoopMode mode = CFRunLoopCopyCurrentMode(CFRunLoopGetCurrent());
                NSLog(@"kCFRunLoopEntry - %@", mode);
                CFRelease(mode);
                break;
            }
                
            case kCFRunLoopExit: {
                CFRunLoopMode mode = CFRunLoopCopyCurrentMode(CFRunLoopGetCurrent());
                NSLog(@"kCFRunLoopExit - %@", mode);
                CFRelease(mode);
                break;
            }
                
            default:
                break;
        }
    });
    // 添加Observer到RunLoop中
    CFRunLoopAddObserver(CFRunLoopGetMain(), observer, kCFRunLoopCommonModes);
    // 释放
    CFRelease(observer);
}

然后开始拖拽textView视图,打印结果如下

image.png

由结果可知,当拖拽时为UITrackingRunLoopMode模式,否则为kCFRunLoopDefaultMode模式


本文主要参考MJ老师的教案,非常感谢MJ老师。


项目连接地址 - RunLoop-底层原理

上一篇下一篇

猜你喜欢

热点阅读