https://juejin.im/post/5aef18b76fb9a07aa34a28e6
https://objccn.io/issue-7-3/
KVC
https://developer.apple.com/library/archive/documentation/Cocoa/Conceptual/KeyValueCoding/BasicPrinciples.html#//apple_ref/doc/uid/20002170-BAJEAIEE
KVC的灵活运用上归结为可以通过字符串(key)获取值
KVC使用
- 动态地取值和设值。利用KVC动态的取值和设值是最基本的用途了。
- 用KVC来访问和修改私有变量。对于类里的私有属性,Objective-C是无法直接访问的,但是KVC是可以的。
- Model和字典转换。这是KVC强大作用的又一次体现,KVC和Objc的runtime组合可以很容易的实现Model和字典的转换。
- 修改一些控件的内部属性。最常用的就是个性化UITextField中的placeHolderText了。
- 操作集合
Apple对KVC的valueForKey:方法作了一些特殊的实现,比如说NSArray和NSSet这样的容器类就实现了这些方法。所以可以用KVC很方便地操作集合。 - 用KVC实现高阶消息传递
当对容器类使用KVC时,valueForKey:将会被传递给容器中的每一个对象,而不是容器本身进行操作。结果会被添加进返回的容器中,这样,开发者可以很方便的操作集合来返回另一个集合。
#import <Foundation/Foundation.h>
int main(int argc, const char * argv[]) {
@autoreleasepool {
NSArray* arrStr = @[@"english",@"franch",@"chinese"];
NSArray* arrCapStr = [arrStr valueForKey:@"capitalizedString"];
for (NSString* str in arrCapStr) {
NSLog(@"%@",str);
}
NSArray* arrCapStrLength = [arrStr valueForKeyPath:@"capitalizedString.length"];
for (NSNumber* length in arrCapStrLength) {
NSLog(@"%ld",(long)length.integerValue);
}
}
return 0;
}
打印结果:
2018-05-05 17:16:21.975983+0800 KVCKVO[35824:6395514] English
2018-05-05 17:16:21.976296+0800 KVCKVO[35824:6395514] Franch
2018-05-05 17:16:21.976312+0800 KVCKVO[35824:6395514] Chinese
2018-05-05 17:16:21.976508+0800 KVCKVO[35824:6395514] 7
2018-05-05 17:16:21.976533+0800 KVCKVO[35824:6395514] 6
2018-05-05 17:16:21.976550+0800 KVCKVO[35824:6395514] 7
获取属性值
valueForKey:
- 返回由key参数指定的属性的值。如果根据访问者搜索模式中描述的规则无法找到密钥指定的属性,则该对象会向自身发送valueForUndefinedKey:
消息。默认实现valueForUndefinedKey:
引发了一个NSUndefinedKeyException
,但是子类可能会覆盖此行为并更优雅地处理这种情况。valueForKeyPath:
- 返回相对于接收器的指定键路径的值。密钥路径序列中的任何对象都不符合特定密钥的键值编码 - 即,默认实现valueForKey:
无法找到访问者方法 - 接收valueForUndefinedKey:
消息。dictionaryWithValuesForKeys:
- 返回相对于接收器的键数组的值。该方法调用valueForKey:
数组中的每个键。返回的NSDictionary
包含数组中所有键的值。
设置属性值
-
setValue:forKey:
- 将指定键的值相对于接收消息的对象设置为给定值。setValue:forKey:
自动解包的默认实现NSNumber
和NSValue
表示标量和结构的对象,并将它们分配给属性。有关包装和解包语义的详细信息,请参阅表示非对象值。如果指定的键对应于接收setter调用的对象没有的属性,则该对象会向自身发送
setValue:forUndefinedKey:
消息。默认实现setValue:forUndefinedKey:
引发了NSUndefinedKeyException
。但是,子类可以重写此方法以自定义方式处理请求。 setValue:forKeyPath:
- 设置相对于接收器的指定键路径的给定值。密钥路径序列中不符合特定密钥的密钥值编码的任何对象都接收setValue:forUndefinedKey:
消息。setValuesForKeysWithDictionary:
- 使用字典键标识属性,使用指定字典中的值设置接收器的属性。默认实现调用setValue:forKey:
为每个键-值对,用nil
于NSNull
根据需要的对象。
访问集合属性
mutableArrayValueForKey:
和mutableArrayValueForKeyPath:
它们返回一个行为类似于NSMutableArray
对象的代理对象。mutableSetValueForKey:
和mutableSetValueForKeyPath:
它们返回一个行为类似于NSMutableSet
对象的代理对象。mutableOrderedSetValueForKey:
和mutableOrderedSetValueForKeyPath:
它们返回一个行为类似于NSMutableOrderedSet
对象的代理对象。
集合运算符
当你对一个对象(集合/数组)发送了valueForKeyPath:消息的时候,集合操作允许你通过嵌入关键字的形式作出相应的操作. 集合操作符是一个以@开头的特殊字符串. for example: @distinctUnionOfObjects.self 无论什么时候你在key path中看见了@,它都代表了一个特定的集合方法,其结果可以被返回或者链接.
聚合运算符
以某种方式合并集合的对象
-
@count
返回一个值为集合中对象总数的NSNumber对象; -
@avg
首先把集合中的每个对象都转换为double类型,然后计算其平均值,并返回这个平均值的NSNumber对象; -
@max
使用compare:方法来确定最大值,并返回最大值的NSNumber对象.所以为了保证其正常比较,集合中所有的对象都必须支持和另一个对象的比较,保证其可比性; -
@min
原理和@max一样,其返回的是集合中的最小值的NSNumber对象; -
@sum
首先把集合中的每个对象都转换为double类型,然后计算其总和,并返回总和的NSNumber对象;
数组运算符
@unionOfObjects:
指定该运算符时,valueForKeyPath:创建并返回一个数组,该数组包含与右键路径指定的属性对应的集合的所有对象。与@distinctUnionOfObjects不同,不会删除重复的对象。@distinctUnionOfObjects:
指定该运算符时,valueForKeyPath:创建并返回一个数组,该数组包含与右键路径指定的属性对应的集合的不同对象。
嵌套集合运算符(array中包含array,set中包含set)
-
@distinctUnionOfArrays
指定该运算符时,valueForKeyPath:创建并返回一个数组,该数组包含与右键路径指定的属性对应的所有集合的组合的不同对象。 -
@ unionOfArrays
指定该运算符时,valueForKeyPath:创建并返回一个数组,该数组包含与右键路径指定的属性对应的所有集合的组合的所有对象,而不删除重复项。 -
@ distinctUnionOfSets
指定该运算符时,valueForKeyPath:创建并返回一个NSSet对象,该对象包含与右键路径指定的属性对应的所有集合的组合的不同对象。
此运算符的行为与此类似@distinctUnionOfArrays,只是它处理的是一个NSSet包含NSSet对象实例而不是NSArray实例实例的NSArray实例。
KVO
https://developer.apple.com/library/archive/documentation/Cocoa/Conceptual/KeyValueObserving/KeyValueObserving.html#//apple_ref/doc/uid/10000177-BCICJDHA
KVO 是通过 isa-swizzling 实现的。 基本的流程就是编译器自动为被观察对象创造一个派生类,并将被观察对象的isa 指向这个派生类。如果用户注册了对某此目标对象的某一个属性的观察,那么此派生类会重写这个方法,并在其中添加进行通知的代码。
KVO的实现,是对注册的keyPath中自动实现了两个函数,在setter中,自动调用。
- (void)willChangeValueForKey:(NSString *)key
- (void)didChangeValueForKey:(NSString *)key
手动KVO(禁用KVO)
首先,需要手动实现属性的 setter 方法,并在设置操作的前后分别调用 willChangeValueForKey: 和 didChangeValueForKey方法,这两个方法用于通知系统该 key 的属性值即将和已经变更了;
其次,要实现类方法 automaticallyNotifiesObserversForKey,并在其中设置对该 key 不自动发送通知(返回 NO 即可)。这里要注意,对其它非手动实现的 key,要转交给 super 来处理。
如果需要禁用该类KVO的话直接automaticallyNotifiesObserversForKey返回NO,实现属性的 setter 方法,不进行调用willChangeValueForKey: 和 didChangeValueForKey方法。
依赖键
@interface LabColor : NSObject
@property (nonatomic, assign) double redComponent;
@property (nonatomic, assign) double greenComponent;
@property (nonatomic, assign) double blueComponent;
@property (nonatomic, strong, readonly) UIColor *color;
@end
@implementation LabColor
+ (NSSet<NSString *> *)keyPathsForValuesAffectingColor {
return [NSSet setWithObjects:@"redComponent", @"greenComponent", @"blueComponent", nil];
}
@end
//KVO监听color的时候,redComponent, greenComponent, blueComponent有修改也会触发KVO回调