RunLoop实际应用

RunLoop是控制线程生命周期并接收事件进行处理的机制.是iOS事件响应与任务处理最核心的机制.
主线程的RunLoop在应用启动时自动创建,让主程序死循环,保证程序不退出,防止用户不对app做出操作导致退出app.
但这个死循环是一个很特殊的死循环, 它能够在有事情的时候做事情, 没事情做的时候休息待命, 以节省CPU资源, 提高程序性能.
实现省电,流畅,响应速度快,用户体验好

实际应用有:

1.定时器

注册到RunLoop之后,RunLoop会为其在重复的时间点注册好事件执行.

1.1 NSTimer
NSTimer *timer = [NSTimer scheduledTimerWithTimeInterval:1.5 target:self selector:@selector(FunctionName) userInfo:nil repeats:YES];
self.timer = timer;
[[NSRunLoop mainRunLoop] addTimer:self.timer forMode:NSRunLoopCommonModes];
1.2 CADisplayLink
CADisplayLink *link = [CADisplayLink displayLinkWithTarget:self selector:@selector(FunctionName)];    
self.link = link;
[link addToRunLoop:[NSRunLoop mainRunLoop] forMode:NSRunLoopCommonModes];

注意:定时器不使用一定要销毁!
其中NSRunLoopCommonModes包含kCFRunLoopDefaultMode(NSDefaultRunLoopMode)UITrackingRunLoopMode两种模式
保证时钟在两种RunLoop运行模式下都能被监听到.

1.3 GCD

GCD定时器会自动开启一条子线程,子线程中也会自己开启了runloop.自己创建,自己管理,全不用我们手动管理

//首先声明属性
@property (strong, nonatomic) dispatch_source_t timer;
//然后是方法:
- (void)GCDTimer {
    /*
     参数1 : 需要创建的源的种类, timer 也是一种数据源
     参数2,参数3: 在你创建timer的时候直接传0就可以了
     参数4: timer触发的代码运行的队列
     */
    dispatch_source_t timer = dispatch_source_create(DISPATCH_SOURCE_TYPE_TIMER, 0, 0, dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0));
    /*
     参数1 : timer定时器
     参数2 : 从什么时间开始触发定时器, DISPATCH_TIME_NOW代表现在
     参数3 : 时间间隔
     参数4 : 表示精度, 传0表示绝对精准
     */
    dispatch_source_set_timer(timer, DISPATCH_TIME_NOW, 2.0 * NSEC_PER_SEC, 0 * NSEC_PER_SEC);
    
    //封装timer需要触发的方法
    dispatch_source_set_event_handler(timer, ^{
        //当前线程是子线程
        NSLog(@"GCDTimer-----%@",[NSThread currentThread]);
        NSLog(@"每两秒打印一次");
    });

    //启用,默认是停止的
    dispatch_resume(timer);

    //用强指针引用, 防止timer释放
    self.timer = timer;
}

2.自动释放池

自动释放池在什么时候创建?什么时候销毁?
创建: 运行循环检测到事件并启动之后,就会创建自动释放池
销毁: 一次完整的运行循环结束之前,会被销毁

经典错误:

for (long i = 0; i < largeNumber; ++i) 
{
   NSString *str = [NSString stringWithFormat:@"hello ) %ld", i]; 
   str = [str uppercaseString]; 
   str = [str stringByAppendingString:@" ) world"];
}

分析:
代码有问题,但是并不是说 largeNumber 没有定义,这个不是重点
重点是 largeNumber 值我们不知道,如果 largeNumber 值非常大,内存就会很高

解决:
引入自动释放池,提前释放对象

for (long i = 0; i < largeNumber; ++i) 
{
   @autoreleasepool { 
    NSString *str = [NSString stringWithFormat:@"hello ) %ld", i];
    str = [str uppercaseString]; 
    str = [str stringByAppendingString:@" ) world"];
   }
 } 

其中,在循环内添加自动释放池会更佳
循环内的运行速度比循环外的要快
循环外的会有内存峰值,如果峰值太大,会造成程序闪退

如果在开发中,遇到部份内存峰值很高,可以尝试添加自动释放池


3.操纵线程
3.1可以用RunLoop实现将一个函数调用在主线程执行
[[NSRunLoop mainRunLoop] performSelector:@selector(FunctionName) withObject:nil];

此外还有三种方式可以实现:
1>GCD:

dispatch_async( dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
  // 执⾏耗时的异步操作...

  dispatch_async(dispatch_get_main_queue(), ^{
  // 回到主线程,执⾏UI刷新操作

  });
});  

2>NSOperation:

NSOperationQueue *mainQueue = [NSOperationQueue mainQueue];
NSBlockOperation *operation = [NSBlockOperation blockOpertionWithBlock:^{

}];
[mainQueue addOperation:operation];

3>NSThread:

[self performSelector:@selector(FunctionName) onThread:[NSThread mainThread] withObject:nil waitUntilDone:YES modes:nil];
[self performSelectorOnMainThread:@selector(method) withObject:nil];

4>pThread 没用过

pthread_t:线程ID
pthread_attr_t:线程属性
pthread_create():创建一个线程
pthread_exit():终止当前线程
pthread_cancel():中断另外一个线程的运行
pthread_join():阻塞当前的线程,直到另外一个线程运行结束
pthread_attr_init():初始化线程的属性
pthread_attr_setdetachstate():设置脱离状态的属性(决定这个线程在终止时是否可以被结合)
pthread_attr_getdetachstate():获取脱离状态的属性
pthread_attr_destroy():删除线程的属性
pthread_kill():向线程发送一个信号
pthread_equal(): 对两个线程的线程标识号进行比较
pthread_detach(): 分离线程
pthread_self(): 查询线程自身线程标识号
3.2常驻线程

让一个子线程不被销毁, 等待其他线程发来消息, 处理事件
没用过

你可能感兴趣的:(RunLoop实际应用)