前言:KVC,KVO 是iOS中一个比较强大的功能,写一下,总结一下,和大家分享一下。技术活,在实际的工作中,学到的东西,能在最合适的场景应用是最关键的。
一、KVC (Key-value coding)键值编码
1、什么是KVC
就是指iOS的开发中,可以允许开发者通过Key名直接访问对象的属性,或者给对象的属性赋值。而不需要调用明确的存取方法。
这样就可以在 **** 运行时动态地访问和修改对象的属性。而不是在编译时确定,这也是iOS开发中的黑魔法之一。
很多高级的iOS开发技巧都是基于KVC实现的。
2、上代码
#import <UIKit/UIKit.h>
@interface ViewController : UIViewController
@end
/** init Dog **/
@interface Dog : NSObject;
@property (nonatomic, copy) NSString *chineseName;
@end
/** init Animal **/
@interface Animal : NSObject
@property (nonatomic, copy) NSString *name;
@property (nonatomic, assign) NSInteger age;
@property (nonatomic, strong) Dog *twoHappy;
- (void)showHeight;
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
代码区
}
@end
@implementation Dog
@end
@interface Animal ()
@property (nonatomic, assign) NSInteger height;
@end
@implementation Animal
- (instancetype)init
{
self = [super init];
if (self) {
self.twoHappy = [[Dog alloc] init];
}
return self;
}
- (void)showHeight {
NSLog(@"my private property height : %ld", self.height);
}
@end
- 1、 设置属性: setValue forkey
Animal *animal = [[Animal alloc] init];
[animal setValue:@"动物" forKey:@"name"]; /** 寻找属性的顺序 setName _name _isName isName **/
NSLog(@" Animal.name %@",[animal valueForKey:@"name"]);
- 2、 访问私有属性 setValue forKeyPath
[animal setValue:[NSNumber numberWithInt:18] forKey:@"height"];
[animal showHeight];
- 3、 设置二层属性: setValue forKeyPath
[animal setValue:@"二哈" forKeyPath:@"twoHappy.chineseName"];
NSLog(@" Animal.twoHappy.chineseName: %@",[animal valueForKeyPath:@"twoHappy.chineseName"]);
- 4、 修改系统控件属性 (例:UITextField占位字体)
UITextField *textField = [[UITextField alloc] initWithFrame:CGRectMake(100, 100, 200, 40)];
textField.placeholder = @"这是一个测试";
textField.backgroundColor = [UIColor purpleColor];
[textField setValue:[UIColor greenColor] forKeyPath:@"_placeholderLabel.textColor"];
[textField setValue:[UIFont systemFontOfSize:12] forKeyPath:@"_placeholderLabel.font"];
[self.view addSubview:textField];
- 5、 通过数组获取全部属性
NSMutableArray *animalArray = [NSMutableArray arrayWithCapacity:0];
NSArray *dogNameArray = @[@"一哈", @"二哈", @"三哈", @"四哈", @"五哈"];
for (int i = 0; i < 5; i++) {
Animal *animal = [[Animal alloc] init];
animal.age = i + 100;
animal.twoHappy.chineseName = [dogNameArray objectAtIndex:i];
[animalArray addObject:animal];
}
/** 获取数组里面所有属性,并返回数组 **/
NSLog(@"all twoHappy %@", [animalArray valueForKeyPath:@"twoHappy.chineseName"]);
- 6、 valueForKeyPath 其他应用
NSArray *array = @[@1, @2, @3, @4, @10];
[array valueForKeyPath:@"@sum.self"]; // 获取和
[array valueForKeyPath:@"@avg.self"]; // 获取平均值
[array valueForKeyPath:@"@max.self"]; // 获取最大值
[array valueForKeyPath:@"@min.self"]; // 获取最小值
// 剔除重复值
NSArray *array2 = @[@"name", @"w", @"aa", @"jimsa", @"aa"];
[array2 valueForKeyPath:@"@distinctUnionOfObjects.self"];
// 获取数组中字典某一个键的所有值
NSArray *array3 = @[@{@"name": @"cookeee",@"code": @1},
@{@"name": @"jim",@"code": @2},
@{@"name": @"jim",@"code": @1},
@{@"name": @"jbos",@"code": @1}];
NSLog(@"allName : %@", [array3 valueForKeyPath:@"name"]);
二、KVO
1、什么是KVO
KVO提供一种机制,指定一个被观察对象(例如A类),当对象某个属性(例如A中的字符串name)发生更改时,对象会获得通知,并作出相应处理;
就是观察某一个对象的属性是否发生变化。
原理:当观察某对象A时,KVO机制动态创建一个对象A当前类的子类,并为这个新的子类重写了被观察属性keyPath的setter 方法。setter 方法随后负责通知观察对象属性的改变状况。
会创建这个对象的子类,并重写被观察属性keyPath 的Setter 方法。负责观察对象属性的改变。
KVO 的实现依赖于 Objective-C 强大的 Runtime
如果赋值没有通过setter方法或者KVC,而是直接修改属性对应的成员变量,例如:仅调用_name = @"newName",这时是不会触发kvo机制
2、大家常提起的和通知代理之间的区别
- 1、notification比KVO多了发送通知的一步。
两者都是一对多,但是对象之间直接的交互,notification明显得多,需要notificationCenter来做为中间交互。而KVO如我们介绍的,设置观察者->处理属性变化,至于中间通知这一环,则隐秘多了,只留一句“交由系统通知”,具体的可参照以上实现过程的剖析。
- 2、 notification 能监听的更多
notification的优点是监听不局限于属性的变化,还可以对多种多样的状态变化进行监听,监听范围广,例如键盘、前后台等系统通知的使用也更显灵活方便。
- 3、 通知,KVO,delegate 各自特点。
通知和KVO都是一对多,通知需要自己发送,但是监听的范围比较广,所以通知的应用范围也比较广一些, 例如监听键盘,系统通知等。
delegate一般是一对一,而这两个可以一对多。前两个都是负责发送接收通知,剩下的事情由系统处理,所以不用返回值;而delegate 则需要通信的对象通过变量(代理)联系;
3、实现监听
- 1、添加监听
self.animal.age = 1;
[self.animal addObserver:self
forKeyPath:@"age"
options:NSKeyValueObservingOptionOld | NSKeyValueObservingOptionNew
context:nil];
- 2、监听
- (void)observeValueForKeyPath:(NSString *)keyPath
ofObject:(id)object
change:(NSDictionary<NSKeyValueChangeKey,id> *)change
context:(void *)context{
if ([keyPath isEqualToString:@"age"]) {
NSLog(@"老的值 :%@ 新的值:%@",[change valueForKey:@"old"],[change valueForKey:@"new"]);
}
}
4、YYKit 实现 KVO 简单封装
YYKit使用Runtime对KVO进一步的封装,这里就不做详细解释了,详情可见YYKit详细代码。
这里只做一个简单的调用,方便开发者使用KVO。
- 1、添加并监听
[self addObserverBlockForKeyPath:@"name"
block:^(id _Nonnull obj, id _Nullable oldVal, id _Nullable newVal) {
NSLog(@"id %@ old %@ new %@",obj,oldVal,newVal);
}];