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: 方法), 所以其它方法都能够按照设计的预期被转发到被代理的对象中。