KVC Collection Operators

2024-02-18 15:08
文章标签 collection kvc operators

本文主要是介绍KVC Collection Operators,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!


Mattt Thompson撰写、 Candyan翻译

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: Takes the double value of each object in the collection, and returns the average value as an 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非常非常慢,我说的对吗?</troll>

作者 Mattt Thompson
Mattt Thompson

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

翻译者

这篇关于KVC Collection Operators的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/721548

相关文章

Collection List Set Map的区别和联系

Collection List Set Map的区别和联系 这些都代表了Java中的集合,这里主要从其元素是否有序,是否可重复来进行区别记忆,以便恰当地使用,当然还存在同步方面的差异,见上一篇相关文章。 有序否 允许元素重复否 Collection 否 是 List 是 是 Set AbstractSet 否

浅谈PHP5中垃圾回收算法(Garbage Collection)的演化

前言 PHP是一门托管型语言,在PHP编程中程序员不需要手工处理内存资源的分配与释放(使用C编写PHP或Zend扩展除外),这就意味着PHP本身实现了垃圾回收机制(Garbage Collection)。现在如果去PHP官方网站(php.net)可以看到,目前PHP5的两个分支版本PHP5.2和PHP5.3是分别更新的,这是因为许多项目仍然使用5.2版本的PHP,而5.3版本对5.2并不是完

Collection的所有的方法演示

import java.util.ArrayList;import java.util.Collection;import java.util.Iterator;public class TestCollection {/*** @param args* Collection的所有的方法演示* 此程序没有使用泛型,所以可以添加任意类型* 以后如果写到泛型会补充这一方面的内容*/public s

C++可以被重载的操作符Overloadable operators

C++允许绝大多数操作符被重载,也就是重新定义操作符实现的功能,这样它们的行为可以被设计出来以适应所有的数据类型,包括类。 以下是C++可以被重载的操作符(Overloadable operators): //四则运算符+ - * / %+= -= *= /= %=//比较运算符> >= == != //赋值运算符= //位操作

Mybatis的collection只显示一条数据

在mybatis一对多映射中,多的那种表mybatis只查询出一条数据,而直接用sql查询数据库的话有多条数据。这是因为两张表的主键同名,都为id_key。解决方法:使用别名。给映射多的那张表的主键使用别名。 sql语句别名 resultmap映射

K8s系列之:解释Kubernetes Operators

K8s系列之:解释Kubernetes Operators 什么是控制器循环?Kubernetes Operator是如何工作的?如何添加自定义资源自定义资源定义Kubernetes Operators:案例研究 你是否曾想过,Site Reliability Engineering(SRE)团队如何成功地管理复杂应用程序?在Kubernetes生态系统中,答案只有一个:Kuber

JavaScript - First step - Numbers and operators

Types of numbers Integers 整数Floating point numbers 单精度浮点数Doubles 双精度浮点数Binary 二进制Octal 八进制Hexadecimal 十六进制 Arithmetic operators 算术运算符 + 加法- 减法* 乘法/ 除法% 求余** 指数 (次方 5 ** 5 = 5 * 5 * 5 * 5 * 5) Oper

Java集合类之Collection

文章目录 1 准备部分1.1 数据结构1.1.1 数组1.1.2 链表 1.2 集合是什么 2 Collection2.1 特点2.2 常用API2.3 遍历Collection的方法2.3.1 toArray方法2.2.2 iterator方法2.3.3 foreach2.3.4 总结 3 List 接口3.1 内容提要3.2 特点3.3 List的API3.3.1 listIterat

Java集合系列—Collection系列集合遍历方式(详解并附有代码案例)

文章目录 Collection遍历方式18.2.1 迭代器遍历18.2.2 增强for遍历18.2.3 Lambda表达式遍历18.2.4 三种通用遍历方式区别 Collection遍历方式 Collection 是单列集合的顶层接口(祖宗接口),它的功能是全部单列集合都可以继承使用的,所以Collection 里面是一些共性的方法。Collection 所有方法被List

SCALA 集合框架文档翻译-Concrete Mutable Collection Classes

具体可变集合类  英文原文:http://docs.scala-lang.org/overviews/collections/concrete-mutable-collection-classes.html 上一章已经介绍了 Scala 标准库中最常使用的不可变具体集合类。本文将对具体可变集合类进行说明。 Array Buffers         ArrayBuffer 中保存了