RunLoop与多线程的原理和使用

2016-11-09  本文已影响106人  元哥830

RunLoop与多线程是一一对应的,但是线程在创建的时候是没有RunLoop的,如果不获取,会一直没有,必须你主动去获取。但是在线程结束时,RunLoop也跟着销毁了。如果在某个线程对你的RunLoop进行某些操作时,必须在线程结束之前进行操作。在程序中有一个主RunLoop,管理程序的生死,具体在UIApplicationMain中执行

int main(int argc, char * argv[]) {
    @autoreleasepool {
        //程序开始执行
        NSLog(@"--------start---------");
        int result;
        //这里会一直执行,相当死循环
        result = UIApplicationMain(argc, argv, nil, NSStringFromClass([AppDelegate class]));
        //程序结束才会执行
        NSLog(@"--------end----------");
        return result;
    }
}

如何获取RunLoop对象?
iOS提供了两套API对RunLoop进行访问或使用,分别是CFRunLoopRef和NSRunLoop
1.CFRunLoopRef 是在CoreFoundation框架内,提供纯C函数的API,所有api都是线程安全的
2.NSRunLoop 是基于CFRunLoopRef封装的,提供面向对象的API,是线程不安全的

获取方式
1.CFRunLoopRef

CFRunLoopGetCurrent( );
CFRunLoopGetMain( );

2.RunLoop

[NSRunLoop currentRunLoop];
[NSRunLoop mainRunLoop];

//相关的五个类

1、CFRunLoopRef

代表一个runloop对象

2、CFRunLoopModeRef

  • 代表runloop运行模式
    1.一个runloop包含若干个mode,每个mode包含若干个timer/source/observer
    2.每个runloop启动时,只能指定一个mode,如果需要切换mode,只能退出loop,再重新指定一个mode进入
    3.同一时刻只能进行一种模式

3、CFRunLoopSourceRef

1.用来管理所有事件的事件源,包括自定义的事件,系统自带的事件
2.Source有两个版本:Source0 和 Source1
1、Source0:为用户主动触发的事件
2、Source1:通过内核和其他线程相互发送消息

4、CFRunLoopTimerRef

1、基本上说的就是NSTimer,基本用法如下实例标示

5、CFRunLoopObserverRef

1.用来监听RunLoop的状态改变
2.状态列表

    kCFRunLoopEntry         = (1UL << 0), // 即将进入Loop
    kCFRunLoopBeforeTimers  = (1UL << 1), // 即将处理 Timer
    kCFRunLoopBeforeSources = (1UL << 2), // 即将处理 Source
    kCFRunLoopBeforeWaiting = (1UL << 5), // 即将进入休眠
    kCFRunLoopAfterWaiting  = (1UL << 6), // 刚从休眠中唤醒
    kCFRunLoopExit          = (1UL << 7), // 即将退出Loop
    kCFRunLoopAllActivities = 0x0FFFFFFFU //所有状态

一个 RunLoop 有很多 Mode ,一个 Mode 里面有很多得 Source/Timer/Observer ,但是同一时刻只能进行一种模式。


RunLoop内部结构图@2x.png
//在原来使用time的时候,我们是直接这样写的,它是直接添加到RunLoop的DefaultMode模式中去得,
如果我们去滑动scrollview的时候,也就是说我们现在操作的是RunLoop的Tracking,因为在前面我们
并没有把time添加到Tracking中去,那么滑动的时候是不会输出的
 NSTimer *timer = [NSTimer scheduledTimerWithTimeInterval:1 target:self selector:@selector(timerAction:) userInfo:nil repeats:YES];
//UITrackingRunLoopMode 当滑动scrollview时,timer也会调用
 [[NSRunLoop currentRunLoop] addTimer:timer forMode:UITrackingRunLoopMode];

补充:
关于定时器的话是有两种的一个是NSTime,但是它是会受RunLoop的模式所影响的,一个是GCD的定时器,它呢是不受RunLoop的模式所影响的,这里的话留给大家一个引子(GCD的定时器是如何不受RunLoop模式的影响),这个也是公司一般很爱问的一个问题。

- (void)showGCDTimer
{
    //GCD一次延时操作
//    dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(6 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
//        NSLog(@"GCD一次延时操作");
//    });
    //GCD定时器
    //1.获得队列
    dispatch_queue_t queue = dispatch_get_main_queue();
    //2.创建一个定时器
    self.timer2 = dispatch_source_create(DISPATCH_SOURCE_TYPE_TIMER, 0, 0, queue);
    //3.设置定时器的各种属性(启动,间隔时间)
    dispatch_time_t start = dispatch_time(DISPATCH_TIME_NOW, (int64_t)7 * NSEC_PER_SEC);
    uint64_t interval = (uint64_t)(1 * NSEC_PER_SEC);
    dispatch_source_set_timer(self.timer2, start, interval, 0);
    
    //4.设置回调(处理定时任务)
    __block int count = 0;
    dispatch_source_set_event_handler(self.timer2, ^{
        NSLog(@"------->GCD Timer");
        count++;
        if (count == 6) {
            //取消定时器
            dispatch_cancel(self.timer2);
            self.timer2 = nil;
        }
    });
    
    //5.启动定时器
    dispatch_resume(self.timer2);
}
- (void)viewDidLoad {
    [super viewDidLoad];
    
    /*
     第一个参数:指定如何给obsever分配存储空间
     第二个参数:需要监听的类型/kCFRunLoopAllActivities为全部
     第三个参数:是否每次都监听
     第四个参数:优先级
     第五个参数:监听状态改变之后的回调函数
     */
    CFRunLoopObserverRef obsever = CFRunLoopObserverCreateWithHandler(CFAllocatorGetDefault(), kCFRunLoopAllActivities, YES, 0, ^(CFRunLoopObserverRef observer, CFRunLoopActivity activity) {
        //        kCFRunLoopEntry         = (1UL << 0), // 即将进入Loop
        //        kCFRunLoopBeforeTimers  = (1UL << 1), // 即将处理 Timer
        //        kCFRunLoopBeforeSources = (1UL << 2), // 即将处理 Source
        //        kCFRunLoopBeforeWaiting = (1UL << 5), // 即将进入休眠
        //        kCFRunLoopAfterWaiting  = (1UL << 6), // 刚从休眠中唤醒
        //        kCFRunLoopExit          = (1UL << 7), // 即将退出Loop
        //        kCFRunLoopAllActivities = 0x0FFFFFFFU //所有状态
        switch (activity) {
            case kCFRunLoopEntry:
                NSLog(@"即将进入Loop");
                break;
            case kCFRunLoopBeforeTimers:
                NSLog(@"即将处理 Timer");
                break;
            case kCFRunLoopBeforeSources:
                NSLog(@"即将处理 Source");
                break;
            case kCFRunLoopBeforeWaiting:
                NSLog(@"即将进入休眠");
                break;
            case kCFRunLoopAfterWaiting:
                NSLog(@"刚从休眠中唤醒");
                break;
            case kCFRunLoopExit:
                NSLog(@"即将退出Loop");
                break;
            default:
                break;
        }
    });
    //给主线程的RunLoop添加一个观察者
    /*
     第一个参数:需要给那个RunLoop添加观察者
     第二个参数:需要添加的observer
     第三个参数:在那种模式下监听
     */
    CFRunLoopAddObserver(CFRunLoopGetMain(), obsever,kCFRunLoopDefaultMode );
    CFRelease(obsever);
}
上一篇 下一篇

猜你喜欢

热点阅读