YYKit源码分析-YYFPSLabel

作者: dangbo | 来源:发表于2017-03-19 10:31 被阅读656次

    YYFPSLabel 是 YYKit 下的一个小工具,可以在运行的时候显示出当前的 FPS,使用 Core Animation 查看帧率不如直接在应用中查看来的直接。
    注意:屏幕静止的时候,YYFPSLabel 显示 60 FPS,Core Animation 显示 1 FPS

    • VSync 信号通知到 App 内时,App 内部会根据当前状态来做处理,如果 CoreAnimation 有未提交内容,则执行提交操作、如果有CADisplayLink 等用户自定义回调,则触发回调
    • Instruments 查看到的 Core Animation FPS,指的是 CA 提交到 GPU 的频率。App 内容静止时,CA 不需要提交内容,那这里 FPS 就没什么计数了

    核心代码

    _link = [CADisplayLink displayLinkWithTarget:[YYWeakProxy proxyWithTarget:self] selector:@selector(tick:)];
    [_link addToRunLoop:[NSRunLoop mainRunLoop] forMode:NSRunLoopCommonModes];
    // CADisplayLink 刷新执行的函数
    - (void)tick:(CADisplayLink *)link {
        if (_lastTime == 0) {
            _lastTime = link.timestamp;
            return;
        }
        // 计算 fps
        _count++;
        NSTimeInterval delta = link.timestamp - _lastTime;
        // 不够 1s 不处理
        if (delta < 1) return;
        _lastTime = link.timestamp;
        float fps = _count / delta;
        _count = 0;
        // 显示 fps 的值 ...
    }
    

    这里使用了YYWeakProxy来解决 NSTimer 或 CADisplayLink 循环引用的问题,先来分析一下循环引用

    • 为什么要使用 invalidate?

        CADisplayLink *link = [CADisplayLink displayLinkWithTarget:self selector:@selector(tick:)];  
      

      当退出界面的时候 displayLink 不会被释放。displayLink 会强引用 target 对象,NSRunLoop 对象会引用 displayLink 。只有当 invalidate 被调用时,NSRunLoop 对象才会释放对 displayLink 的引用,displayLink 释放对target的引用。

    • 什么情况下不会执行 dealloc 中的 invalidate ?

        @property (nonatomic, strong) CADisplayLink *link;
        self.link = [CADisplayLink displayLinkWithTarget:self selector:@selector(tick:)];
      

      这种写法会造成循环引用: self->displayLink, displayLink->self,当退出界面的时候 displayLink 不会被释放。

    • 不怎么完美的解决方案

        - (void)viewWillDisappear:(BOOL)animated {
            [super viewWillDisappear:animated];
            [self.link invalidate];
        }
      

      缺点:当 controller push 一个新的页面的时候,本身没有释放,displayLink 不应该被释放

    • 巧妙的解决方案--YYWeakProxy
      原理:生成一个临时对象,让 displayLink 强引用这个临时对象,在这个临时对象中弱引用 self
      self-强->displayLink-强->YYWeakProxy-弱->self,没有形成循环引用

    YYWeakProxy

    核心代码

    @property (nonatomic, weak, readonly) id target;
    
    + (instancetype)proxyWithTarget:(id)target {
        return [[YYWeakProxy alloc] initWithTarget:target];
    }
    //将消息接收对象改为 target
    - (id)forwardingTargetForSelector:(SEL)selector {
        return _target;
    }
    //self 对 target 是弱引用,一旦 target 被释放将调用下面两个方法,如果不实现的话会 crash
    - (void)forwardInvocation:(NSInvocation *)invocation {
        void *null = NULL;
        [invocation setReturnValue:&null];
    }
    
    - (NSMethodSignature *)methodSignatureForSelector:(SEL)selector {
        return [NSObject instanceMethodSignatureForSelector:@selector(init)];
    }
    
    • YYWeakProxy 继承自 NSProxy,是 Foundation 框架两大基类之一,实现了 NSObject 协议。
    • NSProxy 做为消息转发的抽象代理类,自身能够处理的方法极少(仅 <NSObject> 接口中定义的部分方法,没有 init 方法,子类必须实现 initWithXXX: forwardInvocation: 和 methodSignatureForSelector: 方法), 所以其它方法都能够按照设计的预期被转发到被代理的对象中。

    参考链接:
    Source Code Learning - YYKit

    相关文章

      网友评论

      • zenon:有点疑问 ,forwardInvocation 里面为什么做 [invocation setReturnValue:&null] 操作,而且也没有进行转发处理? 返回的签名为什么使用 [NSObject instanceMethodSignatureForSelector:@selector(init)] ?

      本文标题:YYKit源码分析-YYFPSLabel

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