查看当前屏幕帧数的小工具,效果如下:

这里是Demo: YYFPSLabel
这里我把这个小工具从 YYKit 中抽出来,在学习大牛的代码的过程中,收货了不少东西,这里做个笔记:
1、FPSLabel 实现思路:
使用 CADisplayLink 的 timestamp 属性,配合 timer 的执行次数计算得出 FPS数,详见代码。
2、NSTimer、CADisplayLink 常见问题:
问题1: UIScrollView 在滑动时,timer 会被暂停的问题。
—— 原因:runloop mode 导致。iOS处理滑动时,mainloop 中UIScrollView的mode是 UITrackingRunLoopMode,会优先保证界面流畅,而 timer 默认的model是 NSDefaultRunLoopMode,所以会出现被暂停。
—— 解决办法:将timer加到 NSRunLoopCommonModes 中。
[_link addToRunLoop:[NSRunLoop mainRunLoop] forMode:NSRunLoopCommonModes];
详见:深入理解RunLoop 一文中关于 定时器 和 RunLoop 的 Mode 的部分
问题2:NSTimer 对于 target 的循环引用问题:
以下代码很常见:
CADisplayLink *_link = [CADisplayLink displayLinkWithTarget:self selector:@selector(tick:)];
[NSTimer timerWithTimeInterval:1.f target:self selector:@selector(tick:) userInfo:nil repeats:YES];
—— 原因:以上两种用法,都会对self强引用,此时 timer持有 self,self 也持有 timer,循环引用导致页面 dismiss 时,双方都无法释放,造成循环引用。
此时使用 __weak 也不能有效解决:
__weak typeof(self) weakSelf = self;
_link = [CADisplayLink displayLinkWithTarget:weakSelf selector:@selector(tick:)];
效果如下:

可以看到 页面 dismiss 后,计时器仍然在打印
—— 解决办法:1、在页面退出前,或者合适的时候,手动停止 timer,结束循环引用。
注意:在 dealloc 方法中是肯定不行的!由于循环引用,dealloc 方法不会进。

——解决办法:2、YYFPSLabel 作者提供的 YYWeakProxy
@interface YYWeakProxy : NSProxy
@end
// 使用方式:
_link = [CADisplayLink displayLinkWithTarget:[YYWeakProxy proxyWithTarget:self] selector:@selector(tick:)];