美文网首页iOS程序猿小知识点好东西
RunLoop解决内存峰值方法记录

RunLoop解决内存峰值方法记录

作者: oldSix_Zhu | 来源:发表于2016-10-03 00:26 被阅读269次

    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解决内存峰值方法记录

        本文链接:https://www.haomeiwen.com/subject/wiidyttx.html