iOS. Приемы программирования - стр. 19
>и lastName, так что если имена наших свойств firstName и lastName
>в будущем изменятся в реализации, нам не придется ничего переделывать
>и наш класс останется работоспособным, поскольку мы сможем просто
>изменить значения этих констант в нашем файле реализации */
>extern NSString *const kFirstNameKey;
>extern NSString *const kLastNameKey;
>@interface Person: NSObject
>@property (nonatomic, copy) NSString *firstName;
>@property (nonatomic, copy) NSString *lastName;
>– (id) objectForKeyedSubscript:(id
>– (void) setObject:(id)paramObject forKeyedSubscript:(id
>@end
Метод objectForKeyedSubscript: будет вызываться в вашем классе всякий раз, когда программист предоставит ключ и захочет прочитать в вашем классе значение, соответствующее данному ключу. Очевидно, тот параметр, который будет вам передан, будет представлять собой ключ, по которому программист хочет считать интересующее его значение. Дополнительно к этому методу мы будем вызывать в нашем классе метод setObject: forKeyedSubscript: всякий раз, когда программист захочет задать значение для конкретного ключа. Итак, в данной реализации мы хотим проверить, ассоциированы ли заданные ключи с именами и фамилиями. Если это так, то собираемся установить/получить в нашем классе значения имени и фамилии:
>#import "Person.h"
>NSString *const kFirstNameKey = @"firstName";
>NSString *const kLastNameKey = @"lastName";
>@implementation Person
>– (id) objectForKeyedSubscript:(id
>NSObject
>if ([keyAsObject isKindOfClass: [NSString class]]){
>NSString *keyAsString = (NSString *)keyAsObject;
>if ([keyAsString isEqualToString: kFirstNameKey] ||
>[keyAsString isEqualToString: kLastNameKey]){
>return [self valueForKey: keyAsString];
>}
>}
>return nil;
>}
>– (void) setObject:(id)paramObject forKeyedSubscript:(id
>NSObject
>if ([keyAsObject isKindOfClass: [NSString class]]){
>NSString *keyAsString = (NSString *)keyAsObject;
>if ([keyAsString isEqualToString: kFirstNameKey] ||
>[keyAsString isEqualToString: kLastNameKey]){
>[self setValue: paramObject forKey: keyAsString];
>}
>}
>}
>@end
Итак, в этом коде мы получаем ключ в методе objectForKeyedSubscript:, а в ответ должны вернуть объект, который ассоциирован в нашем экземпляре с этим ключом. Ключ, который получаем, – это объект, соответствующий протоколу NSCopying. Это означает, что при желании мы можем сделать копию такого объекта. Рассчитываем на то, что ключ будет представлять собой строку, чтобы мы могли сравнить его с готовыми ключами, которые были заранее объявлены в начале класса. В случае совпадения зададим значение данного свойства в этом классе. После этого воспользуемся методом valueForKey:, относящимся к объекту NSObject, чтобы вернуть значение, ассоциированное с заданным ключом. Но, разумеется, прежде, чем так поступить, мы должны гарантировать, что данный ключ – один из тех, которые мы ожидаем. В методе setObject: forKeyedSubscript: мы делаем совершенно противоположное – устанавливаем значения для заданного ключа, а не возвращаем их.
Теперь в любой части вашего приложения вы можете инстанцировать объект типа Person и использовать заранее определенные ключи kFirstNameKey и kLastNameKey, чтобы изменить значения свойств firstName и lastName, вот так: