使用Time Profiler给K线图做的一次优化

Time Profiler(时间分析器)

用来检测app执行过程中每个方法所用的时间


image.png

然后就可以看到每段代码所执行的时间了,点击右侧调用的方法,还可以跳转到代码显示更加直观详细。


image.png

优化部分:
前提是已经使用leaks查看过,没有内存泄漏。另外听说layer是使用GPU的,而且苹果爸爸做了优化,所以将各种绘图转成各种layer:CAShapeLayer、CATextLayer、CAGradientLayer,然而,并无卵用,因为啥呢,因为这里卡顿的原因更多是耗时计算,而不是排版绘图,跟用不用layer并没啥亲戚关系,另外创建layer又是一大片对象,好像有点得不偿失。
由于项目是金融类软件,要命的k线图加上手势,性能差得不要不要的。
通过该工具,可以看到每次滑动屏幕,计算量大概要用1s的时间,我的天,难怪fps是个位数。。。
既然如此耗时,首先想到的是,把计算过程放到子线程中去,效果很明显,fps瞬间飙到了40。可惜依然是不舒服,fps虽然上来了,但是延迟问题比较严重,图像变化总是慢半拍。
检查哪个方法耗时比较多

  • NSDateFormatter
+ (NSString *)formatTimeFrom:(NSDate *)date withType:(YJStockType)type
{
    NSDateFormatter *formatter = [NSDateFormatter new];

    switch (type) {
        case YJStockTypeEachMinute:
            [formatter setDateFormat:@"HH:mm"];
            break;
        case YJStockTypeFiveDay:
            [formatter setDateFormat:@"MM-dd HH:mm"];
            break;
        case YJStockTypeMonth:
            [formatter setDateFormat:@"yyyy-MM"];
            break;
        case YJStockTypeDay:
            [formatter setDateFormat:@"yyyy-MM-dd"];
            break;
        default:
            [formatter setDateFormat:@"yyyy-MM-dd HH:mm"];
            break;
    }
    return [formatter stringFromDate:date];
}

这家伙尽然能用到544ms,厉害厉害。开始想到的是使用YYModel里的做法

/// Parse string to date.
static force_inline NSDate *YYNSDateFromString(__unsafe_unretained NSString *string) {
    typedef NSDate* (^YYNSDateParseBlock)(NSString *string);
    #define kParserNum 34
    static YYNSDateParseBlock blocks[kParserNum + 1] = {0};
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        {
            /*
             2014-01-20  // Google
             */
            NSDateFormatter *formatter = [[NSDateFormatter alloc] init];
            formatter.locale = [[NSLocale alloc] initWithLocaleIdentifier:@"en_US_POSIX"];
            formatter.timeZone = [NSTimeZone timeZoneForSecondsFromGMT:0];
            formatter.dateFormat = @"yyyy-MM-dd";
            blocks[10] = ^(NSString *string) { return [formatter dateFromString:string]; };
        }
        
        {
            /*
             2014-01-20 12:24:48
             2014-01-20T12:24:48   // Google
             2014-01-20 12:24:48.000
             2014-01-20T12:24:48.000
             */
            NSDateFormatter *formatter1 = [[NSDateFormatter alloc] init];
            formatter1.locale = [[NSLocale alloc] initWithLocaleIdentifier:@"en_US_POSIX"];
            formatter1.timeZone = [NSTimeZone timeZoneForSecondsFromGMT:0];
            formatter1.dateFormat = @"yyyy-MM-dd'T'HH:mm:ss";
            
            NSDateFormatter *formatter2 = [[NSDateFormatter alloc] init];
            formatter2.locale = [[NSLocale alloc] initWithLocaleIdentifier:@"en_US_POSIX"];
            formatter2.timeZone = [NSTimeZone timeZoneForSecondsFromGMT:0];
            formatter2.dateFormat = @"yyyy-MM-dd HH:mm:ss";

            NSDateFormatter *formatter3 = [[NSDateFormatter alloc] init];
            formatter3.locale = [[NSLocale alloc] initWithLocaleIdentifier:@"en_US_POSIX"];
            formatter3.timeZone = [NSTimeZone timeZoneForSecondsFromGMT:0];
            formatter3.dateFormat = @"yyyy-MM-dd'T'HH:mm:ss.SSS";

            NSDateFormatter *formatter4 = [[NSDateFormatter alloc] init];
            formatter4.locale = [[NSLocale alloc] initWithLocaleIdentifier:@"en_US_POSIX"];
            formatter4.timeZone = [NSTimeZone timeZoneForSecondsFromGMT:0];
            formatter4.dateFormat = @"yyyy-MM-dd HH:mm:ss.SSS";
            
            blocks[19] = ^(NSString *string) {
                if ([string characterAtIndex:10] == 'T') {
                    return [formatter1 dateFromString:string];
                } else {
                    return [formatter2 dateFromString:string];
                }
            };

            blocks[23] = ^(NSString *string) {
                if ([string characterAtIndex:10] == 'T') {
                    return [formatter3 dateFromString:string];
                } else {
                    return [formatter4 dateFromString:string];
                }
            };
        }
        
        {
            /*
             2014-01-20T12:24:48Z        // Github, Apple
             2014-01-20T12:24:48+0800    // Facebook
             2014-01-20T12:24:48+12:00   // Google
             2014-01-20T12:24:48.000Z
             2014-01-20T12:24:48.000+0800
             2014-01-20T12:24:48.000+12:00
             */
            NSDateFormatter *formatter = [NSDateFormatter new];
            formatter.locale = [[NSLocale alloc] initWithLocaleIdentifier:@"en_US_POSIX"];
            formatter.dateFormat = @"yyyy-MM-dd'T'HH:mm:ssZ";

            NSDateFormatter *formatter2 = [NSDateFormatter new];
            formatter2.locale = [[NSLocale alloc] initWithLocaleIdentifier:@"en_US_POSIX"];
            formatter2.dateFormat = @"yyyy-MM-dd'T'HH:mm:ss.SSSZ";

            blocks[20] = ^(NSString *string) { return [formatter dateFromString:string]; };
            blocks[24] = ^(NSString *string) { return [formatter dateFromString:string]?: [formatter2 dateFromString:string]; };
            blocks[25] = ^(NSString *string) { return [formatter dateFromString:string]; };
            blocks[28] = ^(NSString *string) { return [formatter2 dateFromString:string]; };
            blocks[29] = ^(NSString *string) { return [formatter2 dateFromString:string]; };
        }
        
        {
            /*
             Fri Sep 04 00:12:21 +0800 2015 // Weibo, Twitter
             Fri Sep 04 00:12:21.000 +0800 2015
             */
            NSDateFormatter *formatter = [NSDateFormatter new];
            formatter.locale = [[NSLocale alloc] initWithLocaleIdentifier:@"en_US_POSIX"];
            formatter.dateFormat = @"EEE MMM dd HH:mm:ss Z yyyy";

            NSDateFormatter *formatter2 = [NSDateFormatter new];
            formatter2.locale = [[NSLocale alloc] initWithLocaleIdentifier:@"en_US_POSIX"];
            formatter2.dateFormat = @"EEE MMM dd HH:mm:ss.SSS Z yyyy";

            blocks[30] = ^(NSString *string) { return [formatter dateFromString:string]; };
            blocks[34] = ^(NSString *string) { return [formatter2 dateFromString:string]; };
        }
    });
    if (!string) return nil;
    if (string.length > kParserNum) return nil;
    YYNSDateParseBlock parser = blocks[string.length];
    if (!parser) return nil;
    return parser(string);
    #undef kParserNum
}

后来一想人家这是为了尽量满足各种奇葩后台而准备的,自己的没有这么复杂的需求,干脆一点用内存代替计算,在模型中添加一个字典

@property (nonatomic, strong) NSDate *time;

@property (nonatomic, assign, readonly) NSInteger month;
@property (nonatomic, assign, readonly) NSInteger year;
@property (nonatomic, assign, readonly) NSInteger day;
@property (nonatomic, assign, readonly) NSInteger week;
@property (nonatomic, assign, readonly) NSInteger hour;
@property (nonatomic, assign, readonly) NSInteger minute;
@property (nonatomic, assign, readonly) NSInteger second;
  
@property (nonatomic, strong) NSMutableDictionary<NSNumber *, NSString *> *formatTimes;

重写setTime方法,在获取到不同的time的时候,为各个时间组件赋值,这些本来业务逻辑也会用到。

- (void)setTime:(NSDate *)time
{
    _time = time;
    {
        NSCalendar *canlendar = [NSCalendar currentCalendar];
        NSDateComponents *components = [canlendar components:NSCalendarUnitYear | NSCalendarUnitMonth | NSCalendarUnitDay | NSCalendarUnitWeekday | NSCalendarUnitHour | NSCalendarUnitMinute | NSCalendarUnitSecond fromDate:time];
        _year = components.year;
        _month = components.month;
        _day = components.day;
        _week = components.weekday;
        _hour = components.hour;
        _minute = components.minute;
        _second = components.second;
    }
}

使用懒加载的方式添加formatTime

+ (NSString *)formatTimeFrom:(YJStockModel *)stock withType:(YJStockType)type
{
    NSString *time = stock.formatTimes[@(type)];
    if (time) return time;
    
    switch (type) {
        case YJStockTypeEachMinute:
        time = [NSString stringWithFormat:@"%02zd:%02zd", stock.hour, stock.minute];
        break;
        case YJStockTypeFiveDay:
        time = [NSString stringWithFormat:@"%02zd-%02zd %02zd:%02zd", stock.month, stock.day, stock.hour, stock.minute];
        break;
        case YJStockTypeMonth:
        time = [NSString stringWithFormat:@"%zd-%02zd", stock.year, stock.month];
        break;
        case YJStockTypeDay:
        time = [NSString stringWithFormat:@"%zd-%02zd-%02zd", stock.year, stock.month, stock.day];
        break;
        default:
        time = [NSString stringWithFormat:@"%zd-%02zd-%02zd %02zd:%02zd", stock.year, stock.month, stock.day, stock.hour, stock.minute];
        break;
    }
    stock.formatTimes[@(type)] = time;
    return time;
}

第一次使用时候最多也就耗时几毫秒,后续使用几乎是不耗时的,一下子舒坦好多。

  • 通过几个模型计算而得的值
    比如MA10均线,每次都要从该模型开始,往前拿10个元素,然后计算收盘价的平均值,嗯,麻烦,还是老套路,内存换时间,同上。

  • 减少循环次数
    起初为了传说中的高内聚低耦合,把整个视图切成若干个部分,每个部分单独计算。这就造成了重复遍历数据模型,因为模型数据多到有几百个数据,多次重复遍历也是不好的,所以将遍历放到了最外面,遍历的时候再各个击破,也不算违背高内聚低耦合的思想吧。

  • 复用已经创建好的对象
    尽量避免对象的重复创建,如果已经创建的对象可以拿来复用,那自然是会好些。

  • 避免重复计算
    之前已经计算好的数据,在新数据来的时候,先判断前面的是否需要重新计算,如果不需要,则只计算新来的数据,拼接到之前的即可。

通过上面的一系列操作之后,fps可以保持在55以上,虽然还有很多细节可以优化,懒惰的我已经对这个数值相当满意了,毕竟一开始是个位数。。。
补充一点,测试真机为5s,现在来说应该属于低端机了,4系列这年月果断抛弃了,其实5系列都可以抛弃,优化以后,在7上fps是60,偶尔59,应该可以了吧,想当顺滑了!

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

推荐阅读更多精彩内容