像Ruby,JavaScript,Python等爱好者经常会嘲笑Objective-C那臃肿的语法。
实际上,一门语言是否优雅归结起来就是其怎么样能更好的避免循环。for
,while
语句是一种拖累;即使是快速枚举也一样。无论你怎么样使他们看起来更加的友好,循环依然是一个在自然语言中用非常简单方式描述所做事情的代码块。
“给我这个列表里面所有员工的平均薪酬”,等等。。。
double totalSalary = 0.0;
for(Employee *employee in employees){
totalSalary += employee.salary;
}
double averageSalary = totalSalary / employees.count;
庆幸的是,IOS中的键值编码给了我们更简洁的语法,来处理这件事情:
[employee valueForKeyPath:@"@avg.salary"];
KVC 集合运算符允许在valueForKeyPath:
方法中使用 key path 符号在一个集合中执行方法。无论什么时候你在 key path 中看见了@
,它都代表了一个特定的集合方法,其结果可以被返回或者链接,就像其他的 key path 一样。
集合运算符会根据其返回值的不同分为以下三种类型:
- 简单的集合运算符 返回的是 strings, number, 或者 dates
- 对象运算符 返回的是一个数组
- 数组和集合运算符 返回的是一个数组或者集合
要理解其工作原理,最好方式就是去 action 里面看看。想象一个Product
类和一个由以下数据所组成的products
数组:
@interface Product : NSObject
@property (nonatomic,copy)NSString *name; //名称
@property (nonatomic,assign)double price; //价格
@property (nonatomic,strong)NSDate *launchedOn; //上市时间
@end
键-值 编码会在必要的时候把基本数据类型的数据自动装箱和拆箱到NSNumber
或者NSValue
中来确保一切工作正常。
简单集合操作符
@count
: 返回一个值为集合中对象总数的NSNumber
对象。@sum
: 首先把集合中的每个对象都转换为double
类型,然后计算其总,最后返回一个值为这个总和的NSNumber
对象。@avg
: 把集合中的每个对象都转换为double
类型,返回一个值为平均值的NSNumber
对象。@max
: 使用compare:
方法来确定最大值。所以为了让其正常工作,集合中所有的对象都必须支持和另一个对象的比较。@min
: 和@max
一样,但是返回的是集合中的最小值。
例如:
[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
//提示:你可以简单的通过把 self 作为操作符后面的 key path 来获取一个由NSNumber组成的数组或者集合的总值,例如[@[@(1), @(2), @(3)] valueForKeyPath:@"@max.self"]
对象操作符
想象下,我们有一个inventory
数组,代表了当地苹果商店的当前库存(iPad Mini 不足,并且没有新的 iMac,因为还没有发货):
NSArray *inventory = @["iPhone 5", "iPhone 5", "iPhone 5", "iPad Mini", "MacBook Pro", "MacBook Pro"];
@unionOfObjects
/ @distinctUnionOfObjects
: 返回一个由操作符右边的 key path 所指定的对象属性组成的数组。其中@distinctUnionOfObjects
会对数组去重, 而且 @unionOfObjects
不会.
例如:
[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"
数组和集合操作符
数组和集合操作符跟对象操作符很相似,只不过它是在NSArray
和NSSet
所组成的集合中工作的。如果我们做一些例如:比较几个商店中的库存(和我们上一节类似的appleStore库存
和买 iPhone 5 和 iPad Mini 的versizonStore库存
)这样的工作,这个就会很有用。
-
@distinctUnionOfArrays
/@unionOfArrays
: 返回了一个数组,其中包含这个集合中每个数组对于这个操作符右面指定的 key path 进行操作之后的值。正如你期望的,distinct
版本会移除重复的值。 -
@distinctUnionOfSets
: 和@distinctUnionOfArrays
差不多, 但是它期望的是一个包含着NSSet
对象的NSSet
,并且会返回一个NSSet
对象。因为集合不能包含重复的值,所以它只有distinct
操作。
例如:
NSArray *appleStoreInventory = @[@"iPhone 5", @"iPhone 5s", @"iPhone 6",@"iPhone X",@"iPhone XS",@"iPhone XR",@"iPhone XS Max"];
NSArray *storeInventory = @[@"iPhone XS",@"iPhone XR",@"iPhone XS Max"];
//"iPhone XS Max","iPhone XS","iPhone 5s","iPhone 6","iPhone X","iPhone 5","iPhone XR"
NSLog(@"数组操作-去重:%@",[@[appleStoreInventory,storeInventory] valueForKeyPath:@"@distinctUnionOfArrays.self"]);
//"iPhone 5","iPhone 5s","iPhone 6","iPhone X","iPhone XS","iPhone XR","iPhone XS Max","iPhone XS","iPhone XR","iPhone XS Max"
NSLog(@"数组操作-不去重:%@",[@[appleStoreInventory,storeInventory] valueForKeyPath:@"@unionOfArrays.self"]);
NSSet *appleStoreInventory1 = [NSSet setWithArray:appleStoreInventory];
NSSet *storeInventory1 = [NSSet setWithArray:storeInventory];
//"iPhone XR","iPhone 5","iPhone XS","iPhone XS Max","iPhone 5s","iPhone 6","iPhone X"
NSLog(@"集合操作-去重:%@",[[NSSet setWithArray:@[appleStoreInventory1,storeInventory1]] valueForKeyPath:@"@distinctUnionOfSets.self"]);
支持自定义么?这可能是一个可怕的想法
然而,事实证明,在我们的小伙伴objc/runtime
的帮助下,这个实际上 是 有可以能的实现的。
Guy English有一篇很神奇的文章,在文章中,他swizzles valueForKeyPath:
来解析自定义的DSL,其扩展了一些有趣的效果:
NSArray *names = [allEmployees valueForKeyPath: @"[collect].{daysOff<10}.name"];
这是一个好的想法吗?可能不是。(NSPredicate
更加合适,并且其使得逻辑更加简单,易懂)
KVC 集合运算符是一个想节省几行代码并在这一过程中看起来很酷的人必须要了解的。当像 Ruby 这样的脚本语言自夸它的单行能力是多么的灵活时,我们也许应该花一点儿时间来庆祝 Objective-C 中的约束和集合操作符。