YYKit源码分析-YYFPSLabel

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 做为消息转发的抽象代理类,自身能够处理的方法极少(仅 接口中定义的部分方法,没有 init 方法,子类必须实现 initWithXXX: forwardInvocation: 和 methodSignatureForSelector: 方法), 所以其它方法都能够按照设计的预期被转发到被代理的对象中。

参考链接:
Source Code Learning - YYKit

你可能感兴趣的:(YYKit源码分析-YYFPSLabel)