FHHFPSIndicator-iOS开发显示FPS值

前言


有时在iOS开发过程中我们要检验iOS界面的流畅性,可能会需要显示当前界面的FPS值来作参考以便对界面进行优化。
FHHFPSIndicator提供了显示当前界面的FPS值的功能。仅需要调用:
[FHHFPSIndicator sharedFPSIndicator] show]
就可以在APP的各个界面查看当前FPS值。GitHub链接请戳我
从这篇文章你能读到:

  • FHHFPSIndicator的实现原理简介
  • Demo中的自定义导航栏对RunTime的简单使用

先来看看Demo图


FHHFPSIndicator提供了三种位置的显示,分别是‘中偏下’
、‘中偏左’、‘中偏右’

iPhone4、5系列手机由于顶部状态栏宽度较小,当开启的服务和功能过多的时候可能会导致FPS值和服务描述重合,建议采用‘中偏下’显示。

中偏下
中偏左
中偏右

</br>

使用方法

在AppDelegate.m加入以下代码即可全局显示:

#if defined(DEBUG) || defined(_DEBUG)
#import "FHHFPSIndicator.h"
#endif

@interface AppDelegate ()

@end

@implementation AppDelegate

- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
// Override point for customization after application launch.    
      self.window = [[UIWindow alloc] initWithFrame:[UIScreen mainScreen].bounds];
      [self.window makeKeyAndVisible];

      // Add the follwing code after the window become keywindow
      #if defined(DEBUG) || defined(_DEBUG)
        [[FHHFPSIndicator sharedFPSIndicator] show];
  //        [FHHFPSIndicator sharedFPSIndicator].fpsLabelPosition = FPSIndicatorPositionTopRight;
      #endif
  
      self.window.rootViewController = [[UINavigationController alloc] initWithRootViewController:[[HomeViewController alloc] init]];

      return YES;
  }

注意:需要在指定的window makeKeyAndVisible之后再调用show方法。
隐藏PFS值

#if defined(DEBUG) || defined(_DEBUG)
   [[FHHFPSIndicator sharedFPSIndicator] hide];
#endif

注意:hide后FPS值则全局隐藏,需要再次调用show方法来激活显示。

FHHFPSIndicator的实现原理简介

此处只介绍FPS值如何显示在界面,并且切换了控制器后任然可以正常显示。至于FPS的计算很多文章都有描述,这里就不做介绍,感兴趣的可以看看源码。</br>

FHHFPSIndicator.m:

在初始化方法中创建一个_fpsLabel 的Label用于展示FPS值:

- (id)init {
    if (self = [super init]) {
    _displayLink = [CADisplayLink displayLinkWithTarget:self selector:@selector(displayLinkTick:)];
    [_displayLink setPaused:YES];
    [_displayLink addToRunLoop:[NSRunLoop currentRunLoop] forMode:NSRunLoopCommonModes];
    
    // Create fpsLabel
    _fpsLabel = [[UILabel alloc] init];
    self.fpsLabelPosition = FPSIndicatorPositionBottomCenter;
    _fpsLabel.tag = TAG_fpsLabel;
    
    // Set style for fpsLabel
    [self configFPSLabel];
    
    [[NSNotificationCenter defaultCenter] addObserver: self
                                             selector: @selector(applicationDidBecomeActiveNotification)
                                                 name: UIApplicationDidBecomeActiveNotification
                                               object: nil];
    
    [[NSNotificationCenter defaultCenter] addObserver: self
                                             selector: @selector(applicationWillResignActiveNotification)
                                                 name: UIApplicationWillResignActiveNotification
                                               object: nil];
    
     }
     return self;
}

show方法中,向keyWindow添加了_fpsLabel用于在所有界面中都能显示_fpsLabel。

- (void)show {
    UIWindow *keyWindow = [UIApplication sharedApplication].keyWindow;
    for (UIView *label in keyWindow.subviews) {
        if ([label isKindOfClass:[UILabel class]]&& label.tag == TAG_fpsLabel) {
            return;
        }
    }
    [_displayLink setPaused:NO];
    [keyWindow addSubview:_fpsLabel];
}

踩的坑

单单以上代码虽然能显示出_fpsLabel,但是当切换根控制器的时候(window.rootViewController changed),_fpsLabel会被放置在最底层从而无法显示,这是你的window在执行layoutSubviews的时候系统自动做的处理。

填坑

针对这个问题,写一个分类UIWindow+FHH.m,重写window的layoutSubviews方法,把_fpsLabel置顶层即可。
关键代码:UIWindow+FHH.m

#import "UIWindow+FHH.h"

static NSInteger kFpsLabelTag = 110213;

@implementation UIWindow (FHH)

- (void)layoutSubviews {    
    [super layoutSubviews];    

    for (NSUInteger i = 0; i < self.subviews.count; ++i) {
        UIView *view = self.subviews[self.subviews.count - 1 - i];
        if ([view isKindOfClass:[UILabel class]] && view.tag == kFpsLabelTag) {
            if (view == self.subviews.lastObject) {
                return;
            }
            [self bringSubviewToFront:view];
            return;
        }
     }
}

@end

由于window的layoutSubViews方法可能会被频繁触发,有的同学可能担心会影响性能。这里可以不用担心,在layoutSubViews只是添加一个循环判断,通常情况下如果你不往window添加控件的话window的subviews.count只有2个(_fpsLabel和控制器的wrapperView),循环次数非常低,不会对性能造成影响。
PS:建议在DEBUG模式下使用FHHFPSIndicator

Demo中的导航栏对RunLoop的简单使用

Demo示例中显示的几个控制用到的自定义导航栏是通过Runtime来实现的,这样做的好处是免去的继承的关系,对控制器进行解耦
具体做法:
写一个控制器的分类UIViewController+CustomNavigationBar
,利用运行时给该分类添加属性(自定义导航栏、按钮等)和对应的实现方法。
在需要自定义导航的控制器调用:</br>
[self setNavigationBarTitle:@"SubHome" navLeftButtonIcon:@"nav_return" navRightButtonTitle:@"next"]
即可。
** SubHomeViewController.h**

#import <UIKit/UIKit.h>

@interface SubHomeViewController : UIViewController

@end

** SubHomeViewController.m**

#import "SubHomeViewController.h"
#import "UIViewController+CustomNavigationBar.h"
#import "LastHomeViewController.h"
#import "FHHFPSIndicator.h"

@implementation SubHomeViewController

- (void)viewDidLoad {
    [super viewDidLoad];    
    self.view.backgroundColor = RGBColor(160, 160, 160);

    [self setNavigationBarTitle:@"SubHome" navLeftButtonIcon:@"backupIcon" navRightButtonTitle:@"next"];
    [self.navRightButton addTarget:self action:@selector(p_rightButtonDidClick:) forControlEvents:UIControlEventTouchUpInside];

    [FHHFPSIndicator sharedFPSIndicator].fpsLabelPosition = FPSIndicatorPositionTopLeft;
}

- (void)p_rightButtonDidClick:(UIButton *)btn {
  [self.navigationController pushViewController:[[LastHomeViewController alloc] init] animated:YES];
}

UIViewController+CustomNavigationBar的具体实现请看项目Demo。
GitHub地址:https://github.com/jvjishou/FHHFPSIndicator

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,968评论 6 482
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,601评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 153,220评论 0 344
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,416评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,425评论 5 374
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,144评论 1 285
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,432评论 3 401
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,088评论 0 261
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,586评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,028评论 2 325
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,137评论 1 334
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,783评论 4 324
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,343评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,333评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,559评论 1 262
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,595评论 2 355
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,901评论 2 345

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,520评论 25 707
  • 发现 关注 消息 iOS 第三方库、插件、知名博客总结 作者大灰狼的小绵羊哥哥关注 2017.06.26 09:4...
    肇东周阅读 12,029评论 4 62
  • 不知不觉,二月份都过去了,今天是三月份的第一天,也是我农历的生日。从去年毕业到现在,起初在公司也学到了一些东西,但...
    cobs阅读 179评论 2 0
  • 找经络、查穴位在我们公众号首页的右下角有一个“穴位查找”的工具,你点进去就可以了,更可以学到很多不一样的穴位知识。...
    热点榜中榜阅读 17,277评论 0 14