多路委托中的 Nonretain Object:CFArrayCreateMutable

在某些需求下,我们需要建立一个多路委托协议,以满足多处的回调。比如,一个网络请求数据回来,需要通知多个对象使用。实现方案其实很简单,就是把简单的代理调用,变更成一个代理数组,在回调时,遍历数组发送消息即可。

但是要注意,代理是可以被外部释放的,所以除了我们必须要判断:

    if (self.delegate && [self.delegate respondsToSelector:<#(SEL)#>]) {
            ...
    }

之外,针对多路代理,我们还要对这个数组做一个特殊处理。

我们知道,数组可以会使对象的引用计数增加的。为了保持我们 多路代理数组 不会印象其内元素的引用计数,我们需要使用 CFArrayCreateMutable 来创建一个 不会增加引用计数的 数组。

下面直接就是 Nonretaining Array/Dictionary/Set 的代码了:

NSMutableArray * PDCreateNonRetainingMutableArray()
{
    return (__bridge_transfer NSMutableArray *)CFArrayCreateMutable(nil, 0, nil);
}

NSMutableDictionary *PDCreateNonRetainingMutableDictionary()
{
    return (__bridge_transfer NSMutableDictionary *)CFDictionaryCreateMutable(nil, 0, nil, nil);
}

NSMutableSet *PDCreateNonRetainingMutableSet()
{
    return (__bridge_transfer NSMutableSet *)CFSetCreateMutable(nil, 0, nil);
}

方法是 C 函数,当然你也可以封装成 OC 的函数,纯属个人喜好。
看到这里,应该知道,把这个 丢到你们对应的类别中 就可以愉快使用了,当然丢代码里也行,反正我会不接你的锅。

测试用例:

    NSObject *obj = [NSObject new];
    NSLog(@"Retain count is %ld", CFGetRetainCount((__bridge CFTypeRef)obj));

    NSMutableArray *array = @[].mutableCopy;
    [array addObject:obj];
    NSLog(@"Retain count is %ld", CFGetRetainCount((__bridge CFTypeRef)obj));

    NSMutableArray *nonretainingArray = PDCreateNonRetainingMutableArray();
    [nonretainingArray addObject:obj];
    NSLog(@"Retain count is %ld", CFGetRetainCount((__bridge CFTypeRef)obj));

结果:

Retain count is 1
Retain count is 2
Retain count is 2

完美。

7月4日更新:

看到有人提到了 NSPointerArray ,这里也就写上来,为何我没有使用 NSPointerArray。

让我们瞅一眼 NSPointerArray 的使用

NSPointerArray *array = [NSPointerArray weakObjectsPointerArray];
[array addPointer:(__bridge void *)obj];

这货跟上面一样丑。那么使用起来,肯定会进行一轮的封装。
让我们再看看性能:性能检查的宏实现在这里

以下为性能测试

写入测试(每 10^2 次写入为一组参考值,重复 10^5 次取平均值):

- (void)measure
{
    NSObject *obj = [NSObject new];

    PDMeasure(@"NSMutableArray 写入速度", {
        NSMutableArray *mutableArray = @[].mutableCopy;
        for (int i = 0; i < 100; i++)
        {
            [mutableArray addObject:obj];
        }
    })
    
    PDMeasure(@"NSNonretainingMutableArray 写入速度", {
        NSMutableArray *nonretainingArray = PDCreateNonRetainingMutableArray();
        for (int i = 0; i < 100; i++)
        {
            [nonretainingArray addObject:obj];
        }
    })


    PDMeasure(@"NSPointerArray 写入速度", {
        NSPointerArray *pointerArray = [NSPointerArray weakObjectsPointerArray];
        for (int i = 0; i < 100; i++)
        {
            [pointerArray addPointer:(__bridge void *)obj];
        }
    })
}

测试结果输出:

[P] <ViewController|NSMutableArray 写入速度> The average runtime for operation is 10479 ns.
[P] <ViewController|NSNonretainingMutableArray 写入速度> The average runtime for operation is 4822 ns.
[P] <ViewController|NSPointerArray 写入速度> The average runtime for operation is 40685 ns.

读取速度几乎相同,这里就不贴代码和结果了。

以上为性能测试

测试结果是:
写入速度 CFArray 实现的 NonRetainMutableArray 比 NSMutableArray 快约 2 倍,比 NSPointerArray 快约 10 倍。
读取速度几乎相同。

既然都封装后再使用的,就选个性能好的吧。

以上。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 173,638评论 25 708
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 135,001评论 19 139
  • 先看示例图 提前声明:这个demo是仿照之前某个人用Objective-C写的示例,点击圆圈手指效果是用的一个第三...
    船长_阅读 1,404评论 0 5
  • 一场凉意初乍的雨 顺着朱红色的瓦砾 濡湿了檐下的窃窃私语 一片暑气未歇的语 逆着凤凰木的孤寂 捂暖了檐下的怯怯丝雨...
    斐炎凉阅读 260评论 15 11
  • 营销调研课,帮一间新兴的英德茶叶公司做策划。 英德红茶好有名,在国内都是响当当的名字 但这间公司走的是高端品牌路线...
    麦提爽阅读 247评论 0 0