Re:从零开始的RunLoop实践01

Re:从零开始的RunLoop实践

本系列文章,因我在网上看了很多RunLoop的文章之后(先膜拜各路大牛),感觉自己大概懂了,但是说实战一下,又无从下码,本着写不出来代码,会再多理论好比有枪开不出子弹,所以尽量以解决开发中实际问题为出发点,主要以网络上的博客和Github找到的代码为基础(这都是大牛的功劳),总结出用以实战的几个demo,主要为了以后自己使用查找方便(所以此系列提及的理论巨少,基本都是代码,观众老爷也可以直接复制粘贴使用起来),公布在网络上,欢迎各位指出错误,帮助本人及观看文章的大家成长学习。

Re:从零开始的Runloop实践01-线程常驻与销毁

本片博文的代码地址:

https://github.com/zyzhangyu/RunLoopDemo

本小节部分代码取自http://weibo.com/1794363822/CvDEHwuvX?type=comment大牛的Github

强烈推荐同行关注此人

实践目的:保证线程长存,有需求时工作,没有时则休息,属于性能优化的一个方法。

先用最简单的方法实现:

   {    
      _zyThread = [[ZYThread alloc] initWithTarget:self selector:@selector(configRunLoop) object:nil];
        [_zyThread setName:@"章鱼线程"];
        [_zyThread start];
    }```
    

    - (void)configRunLoop{
        
        @autoreleasepool {
            NSRunLoop *runLoop = [NSRunLoop currentRunLoop];
            [runLoop addPort:[NSPort port] forMode:NSDefaultRunLoopMode];
            [runLoop run];
        }
    }

使用`- (void)performSelector:(SEL)aSelector onThread`添加任务到RunLoop当中。示例如下:
```  [self performSelector:@selector(test) onThread:_zyThread withObject:nil waitUntilDone:NO];```
 

以上为最简单的使用RunLoop实现线程常驻的方法,缺点是使用`[runLoop run];`开启的RunLoop,在线程中使用`CFRunLoopStop(runLoopRef);`
也无法销毁。

优点:简单方便
缺点:难以结束
特点:实现了需要线程长存的目标,并且做到了有任务时执行,无任务时休眠的高效率。
可以改进的地方:
- 看不出哪里表现了,有任务时执行,无任务时休眠的效率
- 删除困难,到需要结束时,难以结束

针对以上可以改进的两点,才有了今天的主角—demo1.

核心代码如下:

![Paste_Image.png](http://upload-images.jianshu.io/upload_images/163125-ca3ed38ea5ea8f9d.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)
使用do while循环来作为一个控制RunLoop是否继续的一个开关。

核心代码全貌(30行代码,重写在NSThread的子类下面):

![Paste_Image.png](http://upload-images.jianshu.io/upload_images/163125-d98000b48a13fc92.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)
注释1:
- 获取当前的RunLoop,没有的话会创建一个。
- 创建一个CFRunLoopObserverContext
- 创建一个CFRunLoopObserverRef
 这三个东西都是名词,百度一搜,解释的大把大把。

注释2:为RunLoop添加观察者

注释3:如果当前线程有当前设置的runMode下的事件发生,runloop就会启动,处理对应的事件。如果没有事件发生,runloop就会在每一次   `NSDate distantFuture到来时,启动一次当前线程的runloop.`

只要将控制循环是否继续的Bool类型变量continued设置为No,线程就将会退出。
接下来是证明,Runloop会在有任务时执行,无任务时休眠的部分:

两段代码:

void currentRunLoopObserver(CFRunLoopObserverRef observer, CFRunLoopActivity activity, void *info)
{
NSLog(@"Current thread Run Loop activity: %@", printActivity(activity));
}```

static inline NSString* printActivity(CFRunLoopActivity activity)
 {
     NSString *activityDescription;
     switch (activity) {
         case kCFRunLoopEntry:
             activityDescription = @"kCFRunLoopEntry";
             break;
         case kCFRunLoopBeforeTimers:
             activityDescription = @"kCFRunLoopBeforeTimers";
             break;
         case kCFRunLoopBeforeSources:
             activityDescription = @"kCFRunLoopBeforeSources";
             break;
         case kCFRunLoopBeforeWaiting:
             activityDescription = @"kCFRunLoopBeforeWaiting";
             break;
         case kCFRunLoopAfterWaiting:
             activityDescription = @"kCFRunLoopAfterWaiting";
             break;
         case kCFRunLoopExit:
             activityDescription = @"kCFRunLoopExit";
             break;
         default:
             break;
     }
     return activityDescription;
 }

解释一下,上面代码的作用就是—>在之前给RunLoop添加的观察者,每个 Observer 都包含了一个回调(函数指针),当 RunLoop 的状态发生变化时,观察者就能通过回调接受到这个变化。回调内容是打印观察到的状态,可以观测的时间点有以下几个:

 typedef CF_OPTIONS(CFOptionFlags, CFRunLoopActivity) {
     kCFRunLoopEntry         = (1UL << 0), // 即将进入Loop
     kCFRunLoopBeforeTimers  = (1UL << 1), // 即将处理 Timer
     kCFRunLoopBeforeSources = (1UL << 2), // 即将处理 Source
     kCFRunLoopBeforeWaiting = (1UL << 5), // 即将进入休眠
     kCFRunLoopAfterWaiting  = (1UL << 6), // 刚从休眠中唤醒
     kCFRunLoopExit          = (1UL << 7), // 即将退出Loop
 };

这样根据打印出来的状态就可以确定,RunLoop到底是不是在有任务时执行,无任务时休眠了:

Paste_Image.png

第一个分隔线之前,为RunLoop添加了第一个任务—射击,我们知道

线程和 RunLoop 之间是一一对应的,其关系是保存在一个全局的 Dictionary 里。线程刚创建时并没有 RunLoop,如果你不主动获取,那它一直都不会有。RunLoop 的创建是发生在第一次获取时,RunLoop 的销毁是发生在线程结束时。你只能在一个线程的内部获取其 RunLoop(主线程除外)。
第一个分隔线之前,我们可以看到RunLoop中,执行第一个任务到任务结束后,进入到kCFRunLoopBeforeWaiting一个完整的状态变化过程,这也能说明RunLoop在执行完任务之后,休息了。

第一个和第二个分隔线之间,显示了,RunLoop从休息到唤醒在到休息的过程。

第二个分隔线之后,是RunLoop从唤醒到结束的一个过程。

这个Log可以和这张图互相辅助来看:

Paste_Image.png

本片博文的代码地址:

https://github.com/zyzhangyu/RunLoopDemo

参考资料:
http://weibo.com/1794363822/CvDEHwuvX?type=comment大牛的Github

http://blog.ibireme.com/2015/05/18/runloop/ 此篇为RunLoop写的相当好的一篇 强烈建议大家学习 也希望我早日能写出这种技术博文

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,332评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,508评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,812评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,607评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,728评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,919评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,071评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,802评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,256评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,576评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,712评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,389评论 4 332
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,032评论 3 316
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,798评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,026评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,473评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,606评论 2 350

推荐阅读更多精彩内容