zoukankan      html  css  js  c++  java
  • 方法对象KVO/KVC总结

    本篇文章是一篇关于方法对象的帖子

        

    1.KVC和KVO的概念

        

    1.1 KVC:

        

    NSKeyValueCoding的简称,是一种可以直接通过字符串的名字(key)来拜访类属性的机制,而不是通过调用的Setter、Getter方法拜访。

        

     

        

    1.2 KVO:

        

    NSKeyValueObserving的简称,当指定的对象的属性被修改了,允许对象接收到通知的机制。

        

    2 胪陈

        

     

        

    在很多时候接触到很多地方都有对 KVC,KVO 的描述,但是都是一笔带过。只知道这是Object-C提供的一个不错的机制,可以很好的减少代码。

        

    首先我们先了解下 KVO 的机制,KVO:当指定的对象的属性被修改了,允许对象接收到通知的机制。每当在类中定义一个监听

        

     

        

    如: [self addObserver:self forKeyPath:@"items" options:0 context:contexStr];

        

    当然你还可以监听其他对象的属性变更,如:

        

    [person addObserver:money forKeyPath:@"account" options:0 context:contexStr];

        

    只要当前类中 items 这个属性产生的变更都会触发到以下的方法

        

    - (void)observeValueForKeyPath:(NSString *)keyPath

         ofObject:(id)object

         change:(NSDictionary *)change
    context:(void *)context
    当有属性改变,KVO 会提供主动的消息通知。这样开发人员不须要自己去实现这样的方案:每次属性改变了就发送消息通知。

    SEL sel = sel_get_uid("setValue:forKey:");

    KVC在调用方法setValue的时候

        

    KVO 的优点:

        

        

    这是 KVO 机制提供的最大的优点。因为这个方案已被明肯定义,取得框架级支撑,可以方便地采取。

        

    开发人员不须要添加任何代码,不须要设计自己的视察者模型,直接可以在工程里应用。

        

    其次,KVO 的架构非常的壮大,可以很容易的支撑多个视察者视察同一个属性,以及相干的值。

        

    KVC 的实现分析

        

    KVC 运用了一个 isa-swizzling 技巧。

        

    isa-swizzling 就是类型混合指针机制。KVC 主要通过 isa-swizzling,来实现其内部查找定位的。

        

    isa 指针,就是 is a kind of 的意思,指向维护分宣布的对象的类。该分宣布实际上包含了指向实现类中的方法的指针和其它数据。

        

    如下 KVC 的代码:

        

    [person setValue:@"personName" forKey:@"name"];

        

    就会被编译器处理成:

        

    IMP method = objc_msg_lookup (person->isa,sel);

        

    method(person,sel,@"personName",@"name");

        

    其中:

        每日一道理
    冰心说道:“爱在左,同情在右,走在生命的两旁,随时撒种,随时开花,将这一径长途,点缀得香花弥漫,使穿枝拂叶的行人,踏着荆棘,不觉得痛苦,有泪可落,却不是悲凉。”

        

    SEL数据类型:它是编译器运行Objective-C里的方法的环境参数。

        

    IMP数据类型:他其实就是一个编译器内部实现时候的函数指针。当Objective-C编译器去处理实现一个方法的时候,就会指向一个IMP对象,这个对象是C语言表述的类型。

        

    (1)首先根据方法名找到运行方法的时候所须要的环境参数。

        

    (2)他会从自己isa指针结合环境参数,找到详细的方法实现的接口。

        

    (3)再直接查找得来的详细的方法实现。

        

    这样的话前面介绍的KVO实现就好理解了

        

    当一个对象注册了一个视察者,被视察对象的isa指针被修改的时候,isa指针就会指向一个旁边类,而不是真实的类。

        

    所以isa指针其实不须要指向实例对象真实的类。所以我们的程序最好不要依赖于isa指针。在调用类的方法的时候,最好要明确对象实例的类名。

        

    这样只有当我们调用KVC去拜访key值的时候KVO才会起作用。所以肯定肯定的是,KVO是基于KVC实现的。

        

    下面实例演示一下:

        

    Person.m:

        

     

    -(void)changeName
    {
        name=@"changeName directly";
    }

        

    PersonMonitor.m:

        

     

    //视察者须要实现的方法
    - (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context
    {
        if([keyPath isEqual:@"name"])
        {
            NSLog(@"change happen,old:%@ new:%@",[change objectForKey:NSKeyValueChangeOldKey],[change objectForKey:NSKeyValueChangeNewKey]);
        }
    }

        

    main.m

        

     

    int main(int argc, const char * argv[])
    {
    
        @autoreleasepool {
            
            //测试代码
            
            Person *p =[[Person alloc] init];//监视对象
            
            //视察者对象
            PersonMonitor *pm= [[PersonMonitor alloc]init];
            //Observer KVO收到通知指定的关键路径相对接收器。
            /*
             *pm 对象注册KVO通知。视察者必须实现键值视察方法observeValueForKeyPath。
             *forKeyPath 关键路径,相对接收器,用于视察的属性。这个值不能为零。
             *options NSKeyValueObservingOptions的组合值,指定包含在视察通知的内容。
             *context 恣意的数据传递给anObserver在observeValueForKeyPath。
             */
            [p addObserver:pm forKeyPath:@"name" options:(NSKeyValueObservingOptionNew|NSKeyValueObservingOptionOld) context:nil];
            
            //测试前的数据
            
            NSLog(@"p.name is %@",p.name);
            
            //通过setvalue 的方法,PersonMonitor的监视将被调用
            
            [p setValue:@"name kvc" forKey:@"name"];
            
            //查看设置后的值
            
            NSLog(@"p name get by kvc is %@",[p valueForKey:@"name"]);
            
            //效果和通过setValue 是分歧的
            
            p.name=@"name change by .name=";
            
            //通过person自己的函数来更改 name
            
            [p changeName]; 
            
            //最后一次修改是直接修改,所以没法产生通知。
            
        }
        return 0;
    }

        

    控制台运行结果:

        

     

        2013-05-23 21:57:01.917 KVC_KVOTest[468:303] p.name is (null)

        2013-05-23 21:57:01.919 KVC_KVOTest[468:303] change happen,old:<null> new:name kvc

        2013-05-23 21:57:01.920 KVC_KVOTest[468:303] p name get by kvc is name kvc

        2013-05-23 21:57:01.920 KVC_KVOTest[468:303] change happen,old:name kvc new:name change by .name=

        

    3 结语

        以上是所有内容,希望对大家有所帮助。

        Demo下载地址:http://download.csdn.net/detail/u010013695/5438389

    文章结束给大家分享下程序员的一些笑话语录: 程序员喝酒
      我偶尔采用“木马策略”、“交叉测试”,时间不长就开始“频繁分配释放资源”,“cache”也是免不了的了,
      不过我从不搞“轮巡”,也不会“捕获异常”,更不会“程序异常”,因为我有理性
    克制的。  

    --------------------------------- 原创文章 By
    方法和对象
    ---------------------------------

  • 相关阅读:
    路由器桥接是个什么玩法
    MAC使用小技巧之------用好mac电脑的10个必知的小技巧!
    学习笔记1--响应式网页+Bootstrap起步+全局CSS样式
    mysql运维必会的一些知识点整理
    面试小结1--填空题
    CSS技术实例1-使用CSS计数器实现数值计算小游戏实例页面
    编译8.0
    解决Windows 10 1809 使用管理员权限运行的程序无法浏览网络驱动器的问题
    android sdk
    酷卓教程 明明已经已经有了面具Magisk 确无法正常使用root权限
  • 原文地址:https://www.cnblogs.com/jiangu66/p/3097717.html
Copyright © 2011-2022 走看看