KVC Collection OperatorsMattt Candyan 🚩🌱

Ruby 爱好者总爱嘲笑 Objective-C 臃肿的语法。

尽管新的Object Literals特性让我们的语法瘦了几斤,但那些红头发的恶霸们还总是用他们的单行map和花哨的Symbol#to_proc来嘲讽我们。

实际上,一门语言是否优雅归结起来就是其怎么样能更好的避免循环。forwhile语句是一种拖累;即使是快速枚举也一样。无论你怎么样使他们看起来更加的友好,循环依然是一个在自然语言中用非常简单方式描述所做事情的代码块

“给我这个列表里面所有员工的平均薪酬”,等等。。。

double totalSalary = 0.0;
for (Employee *employee in employees) {
  totalSalary += [employee.salary doubleValue];
}
double averageSalary = totalSalary / [employees count];

╮(╯_╰)╭

幸运的是,键-值编码给我们了一种更加简洁的,几乎像 Ruby 一样的方式来做这件事:

[employees valueForKeyPath:@"@avg.salary"];

KVC 集合运算符允许在valueForKeyPath:方法中使用 key path 符号在一个集合中执行方法。无论什么时候你在 key path 中看见了@,它都代表了一个特定的集合方法,其结果可以被返回或者链接,就像其他的 key path 一样。

集合运算符会根据其返回值的不同分为以下三种类型:

要理解其工作原理,最好方式就是去 action 里面看看。想象一个Product类和一个由以下数据所组成的products数组:

@interface Product : NSObject
@property NSString *name;
@property double price;
@property NSDate *launchedOn;
@end

键-值 编码会在必要的时候把基本数据类型的数据自动装箱和拆箱到NSNumber或者NSValue中来确保一切工作正常。

NamePriceLaunch Date
iPhone 5$199September 21, 2012
iPad Mini$329November 2, 2012
MacBook Pro$1699June 11, 2012
iMac$1299November 2, 2012

简单集合操作符

例如

[products valueForKeyPath:@"@count"]; // 4
[products valueForKeyPath:@"@sum.price"]; // 3526.00
[products valueForKeyPath:@"@avg.price"]; // 881.50
[products valueForKeyPath:@"@max.price"]; // 1699.00
[products valueForKeyPath:@"@min.launchedOn"]; // June 11, 2012

Pro 提示:你可以简单的通过把 self 作为操作符后面的 key path 来获取一个由NSNumber组成的数组或者集合的总值,例如[@[@(1), @(2), @(3)] valueForKeyPath:@"@max.self"] (/感谢 @davandermobile, 来自 Objective Sea)

对象操作符

想象下,我们有一个inventory数组,代表了当地苹果商店的当前库存(iPad Mini 不足,并且没有新的 iMac,因为还没有发货):

NSArray *inventory = @[iPhone5, iPhone5, iPhone5, iPadMini, macBookPro, macBookPro];

例如

[inventory valueForKeyPath:@"@unionOfObjects.name"]; // "iPhone 5", "iPhone 5", "iPhone 5", "iPad Mini", "MacBook Pro", "MacBook Pro"
[inventory valueForKeyPath:@"@distinctUnionOfObjects.name"]; // "iPhone 5", "iPad Mini", "MacBook Pro"

数组和集合操作符

数则和集合操作符跟对象操作符很相似,只不过它是在NSArrayNSSet所组成的集合中工作的。如果我们做一些例如:比较几个商店中的库存(和我们上一节类似的appleStore库存和买 iPhone 5 和 iPad Mini 的versizonStore库存)这样的工作,这个就会很有用。

例如

[@[appleStoreInventory, verizonStoreInventory] valueForKeyPath:@"@distinctUnionOfArrays.name"]; // "iPhone 5", "iPad Mini", "MacBook Pro"

这可能是一个可怕的想法

令人好奇的是,苹果的 KVC 集合操作符文档冒出了下面这个提示:

注意: 目前还不能自定义集合操作符。

这个提示是有意义的,因为大多数人在第一次看到集合运算符时都在想这个。

然而,事实证明,在我们的小伙伴objc/runtime的帮助下,这个实际上 有可以能的实现的。

Guy English有一篇很神奇的文章,在文章中,他swizzles valueForKeyPath:来解析自定义的DSL,其扩展了一些有趣的效果:

NSArray *names = [allEmployees valueForKeyPath: @"[collect].{daysOff<10}.name"];

这段代码可以得到只有休了不足 10 天假期的人的名字(无疑是要提醒他们去休个假吧!)

或者,来看个可笑的极端情况:

NSArray *albumCovers = [records valueForKeyPath:@"[collect].{artist like 'Bon Iver'}.<NSUnarchiveFromDataTransformerName>.albumCoverImageData"];

Ruby 小伙伴们羡慕吧。只用一行就在艺人记录中过滤出来了名字叫"Bon Iver"的艺人,并且用匹配到的专辑的专辑封面的图像数据初始化了一个NSImage对象。

这是一个好的想法吗?可能不是。(NSPredicate更加合适,并且其使得逻辑更加简单,易懂)

这个很酷吗?当然啦!这个聪明的例子展示了 Objective-C DSL 和元编程未来可能的发展方向。


KVC 集合运算符是一个想节省几行代码并在这一过程中看起来很酷的人必须要了解的。当像 Ruby 这样的脚本语言自夸它的单行能力是多么的灵活时,我们也许应该花一点儿时间来庆祝 Objective-C 中的约束和集合操作符。毕竟,Ruby 非常非常慢,我说的对吗?</troll>


除非另有声明,本文采用知识共享「署名-非商业性使用 3.0 中国大陆」许可协议授权。