iOS学习笔记

KVC官方文档学习(二)----KVC基础之访问对象属性

2019-01-30  本文已影响0人  郝嗨森

官方文档

An object typically specifies properties in its interface declaration, and these properties belong to one of several categories:

一个对象通常在interface声明中指定属性,并且这些属性属于几个类别中的一:

  • Attributes. These are simple values, such as a scalars, strings, or Boolean values. Value objects such as NSNumber and other immutable types such as as NSColor are also considered attributes.
  • To-one relationships. These are mutable objects with properties of their own. An object’s properties can change without the object itself changing. For example, a bank account object might have an owner property that is an instance of a Person object, which itself has an address property. The owner’s address may change without changing the owner reference held by the bank account. The bank account’s owner did not change. Only her address did.
  • To-many relationships. These are collection objects. You commonly use an instance of NSArray or NSSet to hold such a collection, although custom collection classes are also possible.

Listing 2中的BankAccount对象演示了每种属性的类型之一。

Listing 2-1Properties of the BankAccount object

@interface BankAccount : NSObject
 
@property (nonatomic) NSNumber* currentBalance;              // An attribute
@property (nonatomic) Person* owner;                         // A to-one relation
@property (nonatomic) NSArray< Transaction* >* transactions; // A to-many relation
 
@end

In order to maintain encapsulation, an object typically provides accessor methods for the properties on its interface. The object’s author may write these methods explicitly or may rely on the compiler to synthesize them automatically. Either way, the author of code using one of these accessors must write the property name into the code before compiling it. The name of the accessor method becomes a static part of the code that is using it. For example, given the bank account object declared in Listing 2-1, the compiler synthesizes a setter that you can invoke for the myAccount instance:

为了保持封装性,对象通常在interface中提供属性的访问器方法。对象的作者可以显式的编写这些方法或者依赖编译器来自动生成它们。无论哪种方式,代码的作者在编译之前必须使用这些访问器之一把属性名称写到代码中去。访问器方法的名称成为使用它的代码的一个静态部分。例如,给定Listing 2-1声明的银行账户对象,编译器可以合成一个myAccount实例可以调用的setter:

[myAccount setCurrentBalance:@(100.0)];

This is direct, but lacks flexibility. A key-value coding compliant object, on the other hand, offers a more general mechanism to access an object’s properties using string identifiers.

这是直接的,但是缺乏灵活性。另一方面,符合KVC的对象提供了一种使用字符串标识符来访问对象属性的更通用的机制。

使用key和key path来标识对象属性(Identifying an Object’s Properties with Keys and Key Paths)

A key is a string that identifies a specific property. Typically, by convention, the key representing a property is the name of the property itself as it appears in code. Keys must use ASCII encoding, may not contain whitespace, and usually begin with a lowercase letter (though there are exceptions, such as the URL property found in many classes).

key是一个标识特定属性的字符串。通常,按照惯例,表示属性的key是属性在代码中显示的属性本身的名称。key必须使用ASCII编码,可能不包含空格,通常以小写字母开头(尽管有例外,例如在很多类中的URL属性)。

Because the BankAccount class in Listing 2-1 is key-value coding compliant, it recognizes the keys owner, currentBalance, and transactions, which are the names of its properties. Instead of calling the setCurrentBalance: method, you can set the value by its key:

因为BankAccount类是符合KVC的,它能识别key的拥有者,currentBalancetransactions时它的属性的名称。你可以通过它的key来设置值,而不用调用setCurrentBalance:

[myAccount setValue:@(100.0) forKey:@"currentBalance"];

In fact, you can set all the properties of the myAccount object with the same method, using different key parameters. Because the parameter is a string, it can be a variable that is manipulated at run-time.

事实上,你可以用相同的方法使用不同的key参数来设置myAccount对象的所有属性。因为参数是一个字符串,它可以是在运行时操作的变量。

A key path is a string of dot-separated keys used to specify a sequence of object properties to traverse. The property of the first key in the sequence is relative to the receiver, and each subsequent key is evaluated relative to the value of the previous property. Key paths are useful for drilling down into a hierarchy of objects with a single method call.

key path是一个用点分隔的key组成的字符串,用于指定要遍历的对象属性序列。序列中的第一个key是相对于接收者的,并且每一个后续的key相对于前一个属性的值进行评估。key paths对使用单个方法调用到向下钻取到对象层次非常有用。

For example, the key path owner.address.street applied to a bank account instance refers to the value of the street string that is stored in the address of the bank account’s owner, assuming the Person and Address classes are also key-value coding compliant.

例如,应用于银行账户实例的key path owner.address.street是指存储在银行账户所有者的地址中的街道字符串的值,假设PersonAddress类都是符合KVC的。

NOTE
In Swift, instead of using a string to indicate a key or key path, you can use a #keyPath expression. This offers the advantage of a compile time check, as described in the Keys and Key Paths section of the Using Swift with Cocoa and Objective-C (Swift 3) guide.

注意:在Swift中你可以使用#keyPath表达式来表示一个key或者key path,而不是用字符串。这提供了编译时检查的有点,如Using Swift with Cocoa和Objective-C (Swift 3) guide中的Keys and Key Paths部分所述。

使用key获取属性值(Getting Attribute Values Using Keys)

An object is key-value coding compliant when it adopts the NSKeyValueCoding protocol. An object that inherits from NSObject, which provides a default implementation of the protocol’s essential methods, automatically adopts this protocol with certain default behaviors. Such an object implements at least the following basic key-based getters:

当对象采用了 NSKeyValueCoding协议它就是符合KVC的。继承自 NSObject(它提供了这个协议的基本方法的默认实现)的对象通过某些默认的行为自动采用这个协议。这样的对象至少实现了以下基本的基于key的getters:

  • valueForKey: - Returns the value of a property named by the key parameter. If the property named by the key cannot be found according to the rules described in Accessor Search Patterns, then the object sends itself a valueForUndefinedKey: message. The default implementation of valueForUndefinedKey: raises an NSUndefinedKeyException, but subclasses may override this behavior and handle the situation more gracefully.
  • valueForKeyPath: - Returns the value for the specified key path relative to the receiver. Any object in the key path sequence that is not key-value coding compliant for a particular key—that is, for which the default implementation of valueForKey: cannot find an accessor method—receives a valueForUndefinedKey: message.
  • dictionaryWithValuesForKeys: - Returns the values for an array of keys relative to the receiver. The method calls valueForKey: for each key in the array. The returned NSDictionary contains values for all the keys in the array.

NOTE
Collection objects, such as NSArray, NSSet, and NSDictionary, can’t contain nil as a value. Instead, you represent nil values using the NSNull object. NSNull provides a single instance that represents the nil value for object properties. The default implementations of dictionaryWithValuesForKeys: and the related setValuesForKeysWithDictionary: translate between NSNull (in the dictionary parameter) and nil (in the stored property) automatically.

注意 集合对象,例如NSArrayNSSetNSDictionary都不能包含nil作为值。相反,可以使用NSNull表示nil值。NSNull提供了一个单独的接口表示对象属性的nildictionaryWithValuesForKeys:和相关的setValuesForKeysWithDictionary:的默认实现在NSNull(在字典参数中)和nil(在存储的属性中)之间自动转换。

When you use a key path to address a property, if any but the final key in the key path is a to-many relationship (that is, it references a collection), the returned value is a collection containing all the values for the keys to the right of the to-many key. For example, requesting the value of the key path transactions.payee returns an array containing all the payee objects for all the transactions. This also works for multiple arrays in the key path. The key path accounts.transactions.payee returns an array with all the payee objects for all the transactions in all the accounts.

当你使用key path来寻址一个属性时,如果key path中的最后一个key是to-many关系(即它引用一个集合),则返回值是一个集合,这个集合包含to-many关系key的右侧所有key的值。例如,请求key path transactions.payee的值返回一个包含所有transactions的所有payee对象的数组。这也适用于key path中的多个数组。key path accounts.transactions.payee返回一个包含所有accounts的所有transactions的所有payee对象的数组。

用键来设置属性值(Setting Attribute Values Using Keys)

As with getters, key-value coding compliant objects also provide a small group of generalized setters with default behavior based upon the implementation of the NSKeyValueCoding protocol found in NSObject:

和getters一样,符合kvc的对象也提供了一小组具有默认行为的通用setter,这些行为基于 NSObject中的NSKeyValueCoding协议的实现:

  • setValue:forKey: - Sets the value of the specified key relative to the object receiving the message to the given value. The default implementation of setValue:forKey: automatically unwraps NSNumber and NSValue objects that represent scalars and structs and assigns them to the property. See Representing Non-Object Values for details on the wrapping and unwrapping semantics.
    If the specified key corresponds to a property that the object receiving the setter call does not have, the object sends itself a setValue:forUndefinedKey: message. The default implementation of setValue:forUndefinedKey: raises an NSUndefinedKeyException. However, subclasses may override this method to handle the request in a custom manner.
  • setValue:forKeyPath: - Sets the given value at the specified key path relative to the receiver. Any object in the key path sequence that is not key-value coding compliant for a particular key receives a setValue:forUndefinedKey: message.
  • setValuesForKeysWithDictionary: - Sets the properties of the receiver with the values in the specified dictionary, using the dictionary keys to identify the properties. The default implementation invokes setValue:forKey: for each key-value pair, substituting nil for NSNull objects as required.

In the default implementation, when you attempt to set a non-object property to a nil value, the key-value coding compliant object sends itself a setNilValueForKey: message. The default implementation of setNilValueForKey: raises an NSInvalidArgumentException, but an object may override this behavior to substitute a default value or a marker value instead, as described in Handling Non-Object Values.

在默认实现中,当你尝试设置一个非对象属性为nil值时,符合KVC的对象会给自身发送一个 setNilValueForKey:消息。setNilValueForKey:的默认实现会引发NSInvalidArgumentException,但是对象可能覆盖这个行为来替换默认值或标记值,如Handling Non-Object Values所述。

使用键简化对象访问(Using Keys to Simplify Object Access)

To see how key-based getters and setters can simplify your code, consider the following example. In macOS, NSTableView and NSOutlineView objects associate an identifier string with each of their columns. If the model object backing the table is not key-value coding compliant, the table’s data source method is forced to examine each column identifier in turn to find the correct property to return, as shown in Listing 2-2. Further, in the future, when you add another property to your model, in this case the Person object, you must also revisit the data source method, adding another condition to test for the new property and return the relevant value.

要想知道基于键的 getters 和 setters 如何简化你的代码,考虑一下以下的例子。在macos中,NSTableViewNSOutlineView对象用一个标识字符串来关联它们的每一列。如果支持表的模型对象不符合KVC,则表的数据源方法将被强制检查每一列的标识以查找要返回的正确属性,如Listing 2-2所展示的一样,在未来,当你给你的模型(在本例中为Person对象)添加另一个属性时,还必须重新访问数据源方法,添加另一个条件来测试新的属性并返回相关的值。

Listing 2-2Implementation of data source method without key-value coding

- (id)tableView:(NSTableView *)tableview objectValueForTableColumn:(id)column row:(NSInteger)row
{
    id result = nil;
    Person *person = [self.people objectAtIndex:row];
 
    if ([[column identifier] isEqualToString:@"name"]) {
        result = [person name];
    } else if ([[column identifier] isEqualToString:@"age"]) {
        result = @([person age]);  // Wrap age, a scalar, as an NSNumber
    } else if ([[column identifier] isEqualToString:@"favoriteColor"]) {
        result = [person favoriteColor];
    } // And so on...
 
    return result;
}

On the other hand, Listing 2-3 shows a much more compact implementation of the same data source method that takes advantage of a key-value coding compliant Person object. Using only the valueForKey: getter, the data source method returns the appropriate value using the column identifier as a key. In addition to being shorter, it is also more general, because it continues to work unchanged when new columns are added later, as long as the column identifiers always match the model object’s property names.

另一方面,Listing 2-3展示了相同数据源方法的更紧凑的实现,该方法利用了符合KVC的Person对象。仅使用 valueForKey:,数据源方法使用列标识符作为key返回合适的值。除了更短,还更通用,因为在以后添加新列时它能继续保持不变,只要列标识符总是和模型对象属性名称匹配即可。

Listing 2-3Implementation of data source method with key-value coding

- (id)tableView:(NSTableView *)tableview objectValueForTableColumn:(id)column row:(NSInteger)row
{
    return [[self.people objectAtIndex:row] valueForKey:[column identifier]];
}
上一篇 下一篇

猜你喜欢

热点阅读