美文网首页我爱编程
KVC、KVO 学习与应用

KVC、KVO 学习与应用

作者: 傻啦啦了 | 来源:发表于2018-05-28 15:16 被阅读0次

    前言: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);
    }];
    

    相关文章

      网友评论

        本文标题:KVC、KVO 学习与应用

        本文链接:https://www.haomeiwen.com/subject/ttrgjftx.html