IOS开发实用技巧(二)

主要学习:
1 . RunLoop深入浅出;
2 . 预缓存思想;
3 . 第三方框架一句代码解决UITableView自动缓存行高;

1.RunLoop初步了解

  • 什么是RunLoop?
    RunLoop就是消息循环,每一个线程内部都有一个消息循环。
    只有主线程的消息循环默认开启,子线程的消息循环默认不开启。

  • RunLoop的目的
    保证程序不退出 。
    负责处理输入事件。
    如果没有事件发生,会让程序进入休眠状态 。

  • 事件类型
    Input Sources (输入源) & Timer Sources (定时源)

  • RunLoop用处
    1 .使用run loop可以使你的线程在有工作的时候工作,没有工作的时候休眠,这可以大大节省系统资源。
    2 .子线程中使用了定时器
    3 .使线程履行周期性任务
    4 .子线程中用了NSURLConnection异步请求,那也需要用到runloop,不然线程退出了,相应的delegate方法就不能触发。
    5 .Cocoa中使用任何performSelector…的方法

2.RunLoop高级用法

  • IOS开发中有两这个对象:NSRunLoop 和 CFRunLoopRef
  • CFRunLoopRef 是在 CoreFoundation 框架内的,它提供了纯 C 函数的 API,所有这些 API 都是线程安全的。
  • NSRunLoop 是基于 CFRunLoopRef 的封装,提供了面向对象的 API,但是这些 API 不是线程安全的。
  • 一般不会用NSRunLoop,因为它不是线程安全的。一般都建议用CFRunLoop,这个是线程安全的。
  • 子线程默认不开启消息循环,主线程默认开启消息循环。
  • [[NSRunLoop currentRunLoop] run]最好写成 CFRunLoopRun();这样写方便我们关闭它。
    CFRunLoopGetCurrent();//获得当前线程的运行循环!
    CFRunLoopStop(CFRunLoopGetCurrent()); //关闭运行循环!

3.demo案例一:

// 创建Observer  
  /*     
     第1个参数: 指定如何给observer分配存储空间     
     第2个参数: 需要监听的状态类型/ kCFRunLoopAllActivities监听所有状态     
     第3个参数: 是否每次都需要监听    
     第4个参数: 优先级     
     第5个参数: 监听到状态改变之后的回调   
  */
    CFRunLoopObserverRef  observer = CFRunLoopObserverCreateWithHandler(CFAllocatorGetDefault(), kCFRunLoopAllActivities, YES, 0, ^(CFRunLoopObserverRef observer, CFRunLoopActivity activity) {
/**
  ypedef 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添加一个观察者    
/* 第1个参数: 需要给哪个RunLoop添加观察者     
     第2个参数: 需要添加的Observer对象     
     第3个参数: 在哪种模式下可以可以监听     
*/
    CFRunLoopAddObserver(CFRunLoopGetMain(), observer, kCFRunLoopDefaultMode);
    // 释放对象
    CFRelease(observer);

4.消息循环即将进入休眠状态时,也就是要空闲时,进行一些后台的操作,这样某种程度上提高流畅度,比如动态缓存cell的行高.

下面的demo是多么的优雅.

    CFRunLoopRef runLoop = CFRunLoopGetCurrent();
    CFRunLoopObserverRef observer = CFRunLoopObserverCreateWithHandler    (kCFAllocatorDefault, kCFRunLoopBeforeWaiting, true, 0, ^(CFRunLoopObserverRef observer, CFRunLoopActivity _) {
        // TODO here
    for (int i = 0;i<2 ;i++ ) {          
              NSLog(@"%d-->%@",i,[NSThread currentThread]);
          }
     });
    CFRunLoopAddObserver(runLoop, observer, kCFRunLoopDefaultMode);
//   CFRunLoopRemoveObserver(runLoop, observer, kCFRunLoopDefaultMode);
//   CFRelease(observer); // 注意释放,否则会造成内存泄露

注意:若要继续深入了解,需要深入Core Foundation框架中observer/timer/source模式.

5.如何用一句话解决缓存行高问题

pod search UITableView+FDTemplateLayoutCell
并导入UITableView+FDTemplateLayoutCell第三方库

#import "UITableView+FDTemplateLayoutCell.h"
- (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath{    return [tableView fd_heightForCellWithIdentifier:@"reuse identifer" configuration:^(id cell) {
        // Configure this cell with data, same as what you've done in "-tableView:cellForRowAtIndexPath:"
       
// Like:模型—数组—赋值

        // cell.entity = self.feedEntities[indexPath.row];
    }];
}

如果要debug调试: self.tableView.fd_debugLogEnabled = YES;
github 地址: https://github.com/forkingdog/UITableView-FDTemplateLayoutCell
天朝的讲解: http://blog.sunnyxx.com/2015/05/17/cell-height-calculation/

总结:学习利用RunLoop空闲时间执行预缓存任务
预缓存机制
利用RunLoop空闲时间执行预缓存任务,预缓存机制将在 UITableView 没有滑动的空闲时刻执行,计算和缓存那些还没有显示到屏幕中的 cell,整个缓存过程完全没有感知,这使得完整列表的高度计算既没有发生在加载时,又没有发生在滑动时,同时保证了加载速度和滑动流畅性.

你可能感兴趣的:(IOS开发实用技巧(二))