美文网首页iOS Developer
给链式语法泼点冷水

给链式语法泼点冷水

作者: alexlee002 | 来源:发表于2017-05-16 00:33 被阅读223次

引言

最近基于block实现的链式语法很火, 不管大牛还是新手, 都搞出好多链式语法的库或者blog介绍。曾经有一段时间, 我也沉迷于此, 去年自己实现的数据库映射框架也用了链式语法: Example。 有些坑, 踩过之后才知道痛, 鉴于目前介绍block链式语法的文章和第三方库, 都没有提到这些问题, 那么, 我就来泼些冷水吧。

问题

0x00

首先是容易导致崩溃, 在我之前的文章【链式语法的坑...】中也提到过,基于block的链式表达式容易出现空指针的崩溃:

FOO *foo = // init foo object here, maybe nil.
foo.say().hello();

在这里, 如果第一行foo为nil, 或者say()返回的对象是nil, 那么, 这里就会崩溃, 原因在前一篇文章中已经论述过, 也提供了相应的解决方案。希望大家在实现或者使用类似代码的时候, 引起注意。

一个比较好的建议是, 参考Masonry, 在一个限定的上下文内使用链式语法。 使用链式语法的对象实例, 是作为参数传递给使用者, 而不是由使用者自行创建, 这样, 我们可以保证使用链式语法的对象, 不会为nil, 避免使用者疏忽导致崩溃:

    [myView mas_makeConstraints:^(MASConstraintMaker *make) {
        make.left.equalTo(self.mas_left).offset(8);
        make.top.equalTo(self.mas_top).offset(8);
        make.height.mas_equalTo(21);
        make.width.mas_equalTo(160);
    }];

// 我们看Masonry源码中, mas_makeConstraints的实现:
// View+MASAdditions.m
- (NSArray *)mas_makeConstraints:(void(^)(MASConstraintMaker *))block {
    self.translatesAutoresizingMaskIntoConstraints = NO;
    MASConstraintMaker *constraintMaker = [[MASConstraintMaker alloc] initWithView:self];
    block(constraintMaker);
    return [constraintMaker install];
}

0x01

性能问题, 相对于上边的问题, 这个显得不那么重要。 但是, 一些调用频繁的方法, 建议尽量不要用基于block实现的链式语法, 因为这将带来一倍多的性能开销, 而可读性/可维护性并没有明显比传统方法实现的链式语法提高很多。

- (MyTestObj *)append:(NSString *)str {
    if (_str == nil) {
        _str = [str mutableCopy];
    } else {
        [_str appendString:str];
    }
    return self;
}

- (void)reset {
    _str = nil;
}

- (MyTestObj *(^)(NSString *str))APPEND {
    return ^ MyTestObj *(NSString *str) {
        if (_str == nil) {
            _str = [str mutableCopy];
        } else {
            [_str appendString:str];
        }
        return self;
    };
}

- (MyTestObj *(^)())RESET {
    return ^ MyTestObj *{
        _str = nil;
        return self;
    };
}

//////////////// test case //////////////////
    MyTestObj *obj = [[MyTestObj alloc] init];
    @autoreleasepool {
        CFTimeInterval t = CFAbsoluteTimeGetCurrent();
        for (NSInteger i = 0; i < 100000; ++i) {
            for (NSInteger j = 0; j < 50; ++j) {
                obj = [obj append:@"conformsToProtocol:"];
            }
            [obj reset];
        }
        CFTimeInterval t1 = CFAbsoluteTimeGetCurrent() - t;
        NSLog(@"method: %fms", t1 * 1000.f);
    }

    obj = [[MyTestObj alloc] init];
    @autoreleasepool {
        obj = [SingletonBase sharedInitance];
        CFTimeInterval t = CFAbsoluteTimeGetCurrent();
        for (NSInteger i = 0; i < 100000; ++i) {
            for (NSInteger j = 0; j < 50; ++j) {
                obj = obj.APPEND(@"conformsToProtocol:");
            }
            obj.RESET();
        }
        NSLog(@"BLOCK: %fms", (CFAbsoluteTimeGetCurrent() - t) * 1000.f);
        [obj destroyInstance];
    }

////////// output: /////////
method: 988.265038ms
BLOCK: 1867.130041ms

可以看出, 使用block的方式,要多消耗一倍的时间。 其实简单分析也可以得出这样的结论, block的方式, 要通过getter方法获取block属性, 然后再执行block方法, 比普通方法多了一个步骤。另外,如果在链式语法某个链条中返回nil, 普通方法实现的链式语法就不会执行后续的方法内部逻辑,但是在block方式中, 我们必须确保每个链条中返回的结果不为nil, 因此block方式的开销会更大。

所以, 一般情况下, 不建议在调用频繁或者需要提升性能的地方使用block链式语法。

Block链式语法的优势?

自行百度(Google)一下。

相关文章

  • 给链式语法泼点冷水

    引言 最近基于block实现的链式语法很火, 不管大牛还是新手, 都搞出好多链式语法的库或者blog介绍。曾经有一...

  • 给自己泼点冷水

    我不是属虎的,可我喜欢吃肉。不光是肉的口感好,吃起来顺口,还因为肉类很有营养。营养专家说,肉类含有丰富的...

  • 给《前任3》泼点冷水

    最近朋友圈被《前任3》刷屏,一般这种片名和海报看起来比较简单粗暴的电影,不看,机缘巧合之下,还是去电影院给它贡献了...

  • 给热点营销泼点冷水

    爆点事件是品牌公关人士最喜闻乐见的事,一旦出现爆点事件,不出一会熟悉的各大品牌就会迅速切入热点、顺势炒作,一场营销...

  • 给小程序泼点冷水

    1月9日的早上,朋友圈被小程序攻陷。 基本上大家都是高唱赞歌,赶地铁上班的路上,大家互相打听,这个小程序不错,那个...

  • 给视频号泼点冷水

    上周写了:聊聊视频号的算法逻辑。大家有学习到什么吗? 要知道你购买的 399 元培训课程是从来不会讲这些东西的,一...

  • 给读书会泼点冷水

    说实话,现在见到市面上一家接一家的读书会,有点想吐的感觉。 泼点冷水。 众所周知知识付费平台续约率不高,知识经济概...

  • 给华为鸿蒙系统泼点冷水

    现在的华为是不能被批评的,只要批评了就是不爱国、不盼着国产科技公司好、就是谷狗果粉崇洋媚外。我相信各位的朋友圈、订...

  • 该给股市泼点冷水了

    本轮A股暴涨,我是从本月一号算起,到前天为止上证指数涨了六百多点,涨幅短短十多天达到了21%多,而创业板指数涨幅竟...

  • 请把冷水泼给自己

    生活中总有那么些人,你买了件新衣服他说为什么不换个颜色,他说他觉得另一个颜色更适合你。你兴致勃勃的跟他分享好玩儿事...

网友评论

    本文标题:给链式语法泼点冷水

    本文链接:https://www.haomeiwen.com/subject/tfekxxtx.html