KVC、KVO 学习与应用

前言:KVC,KVO 是iOS中一个比较强大的功能,写一下,总结一下,和大家分享一下。技术活,在实际的工作中,学到的东西,能在最合适的场景应用是最关键的。

一、KVC (Key-value coding)键值编码

1、什么是KVC

就是指iOS的开发中,可以允许开发者通过Key名直接访问对象的属性,或者给对象的属性赋值。而不需要调用明确的存取方法。
这样就可以在 **** 运行时动态地访问和修改对象的属性。而不是在编译时确定,这也是iOS开发中的黑魔法之一。
很多高级的iOS开发技巧都是基于KVC实现的。

2、上代码

#import 

@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 *)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);
}];

你可能感兴趣的:(KVC、KVO 学习与应用)