KVC 集合运算符

文章转自http://nshipster.cn/
另外iOS 中集合遍历方法的比较和技巧 · sunnyxx的技术博客这篇文章对于KVC 集合运算符的性能也做了描述

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

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

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

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

Objective-C

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

╮(╯_╰)╭

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

Objective-C

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

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

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

  • 简单的集合运算符 返回的是strings, number, 或者 dates
  • 对象运算符 返回的是一个数组
  • 数组和集合运算符 返回的是一个数组或者集合

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

Objective-C

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

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

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

简单集合操作符

  • @count: 返回一个值为集合中对象总数的NSNumber对象。
  • @sum: 首先把集合中的每个对象都转换为double类型,然后计算其总,最后返回一个值为这个总和的NSNumber对象。
  • @avg: 把集合中的每个对象都转换为double类型,返回一个值为平均值的NSNumber对象。
  • @max: 使用compare:方法来确定最大值。所以为了让其正常工作,集合中所有的对象都必须支持和另一个对象的比较。
  • @min: 和@max一样,但是返回的是集合中的最小值。

例如

Objective-C

[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,因为还没有发货):

Objective-C

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

  • @unionOfObjects / @distinctUnionOfObjects: 返回一个由操作符右边的key path所指定的对象属性组成的数组。其中@distinctUnionOfObjects 会对数组去重, 而且 @unionOfObjects 不会.

例如

Objective-C

[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库存)这样的工作,这个就会很有用。

  • @distinctUnionOfArrays / @unionOfArrays: 返回了一个数组,其中包含这个集合中每个数组对于这个操作符右面指定的key path进行操作之后的值。正如你期望的,distinct版本会移除重复的值。

  • @distinctUnionOfSets: 和@distinctUnionOfArrays差不多, 但是它期望的是一个包含着NSSet对象的NSSet,并且会返回一个NSSet对象。因为集合不能包含重复的值,所以它只有distinct操作。

例如

Objective-C

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


这可能是一个可怕的想法

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

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

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

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

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

Objective-C

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

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

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

Objective-C

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

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

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

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


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

作者
Mattt Thompson

Mattt Thompson

Mattt Thompson (@mattt) is the creator & maintainer of AFNetworking and other popular open-source projects, including Postgres.app, ASCIIwwdc and Nomad.

翻译者

Candyan

Candyan 专业 iOS,业余 Android,偶尔捣鼓下Server的工程师。

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

推荐阅读更多精彩内容