Objective-C 中关于KVC\KVO

熟悉oc语法的同学也许都会懂得这么一点:在oc中,类的成员变量或是方法是没有绝对私有的。


私有方法直接通过类实例无法访问,但可以借助oc的“编译运行时”机制,也即“瞎子摸黑”机制(个人理解:只要确定了该类有方法A,管你是私有共有,我用performSelector函数就能调用你),说到这,也许有同学会自然想起,那私有变量如何去访问呢?貌似以前还真没这样搞过,然而现实是可以的,只不过一般我们把变量设为类的私有变量后是不希望自己或是其他人再去访问的,不然我只能说你自己又在找贱了。。。。。。。。。。。。。。


好吧,言归正传,这篇文章主要是想介绍下oc中的KVC、KVO、KVB的实现机制,当然跟我上面说的那些肯定是有关系的啦。

1.KVC:  key-value coding(键值编码)

它 是一种使用字符串标识符,间接访问对象属性的机制,它是很多技术的基础。主要的方法就两对方法:(setValue:forKey,valueForKey)、setValue:forKeyPath,valueForKeyPath);这个东西有什么作用呢,我先不说原理,先说怎么用,例子如下:

@interface A { NSString* foo; }
 ... // 其它代码
 @end
@interface B { NSString* bar; A* myA; }
 ... // 其它代码
@end
@implementation B
... // 假设 A 类型的对象 a,B 类型的对象
 b A* a = ...;
 B* b = ...;
 NSString* s1 = [a valueForKey:@"foo"]; // 正确
 NSString* s2 = [b valueForKey:@"bar"]; // 正确
 NSString* s3 = [b valueForKey:@"myA"]; // 正确
 NSString* s4 = [b valueForKeyPath:@"myA.foo"]; // 正确
 NSString* s5 = [b valueForKey:@"myA.foo"]; // 错误
 NSString* s6 = [b valueForKeyPath:@"bar"]; // 正确
 ...
@end

其实上面说的那两对方法使用上基本是一样的,只是valueForKeyPath的值是一个路径(路径之间以点号 . 分割),比如数据成员就是对象自己,寻值过程就会向下深入下去。
注意,这里的数据成员的名字都是使用的字符串的形式。这种使用方法的最好的用处在于将数据(名字)绑定到一些触发器(尤其是方法调用)上,
例如键值对观察(Key-Value Observing, KVO)等。

上述代码说明了类的成员变量也可以使用基类NSObject的那两对方法去访问,不一定直接通过类实例访问,但是这种方式还是有一定的风险,具体危险情况请参考这个:http://www.devbean.info/2011/04/from_cpp_to_objc_20/
然后我再说下原理,是俺Copy过来的,大家观赏下:
KVC运用了一个isa- swizzling技术。isa-swizzling就是类型混合指针机制。KVC主要通过isa- swizzling,来实现其内部查找定位的。
isa指针,如其名称所指,(就是is a kind of的意思),指向维护分发表的对象的类。该分发表实际上包含了指向实现类中的方法的指针,和其它数据。

比如说如下的一行KVC的代码:

[site setValue:@"sitename" forKey:@"name"];
就会被编译器处理成:

SEL sel = sel_get_uid ("setValue:forKey:");
IMP method = objc_msg_lookup (site->isa,sel);
method(site, sel, @"sitename", @"name");
首先介绍两个基本概念:

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

(2)IMP数据 类型:他其实就是一个编译器内部实现时候的函数指针。当Objective-C编译器去处理实现一个方法的时候,就会指向一个IMP对象,这个对象是C语言表述的类型(事实上,在Objective-C的编译器处理的时候,基本上都是C语言的)。
这下KVC内部的实现就很清楚的清楚了:一个对象在调用setValue的时候,
(1)首先根据方法名找到运行方法的时候所需要的环境参 数。
(2)他会从自己isa指针结合环境参数,找到具体的方法实现的接口。
(3)再直接查找得来的具体的方法实现。

2.KVO: key-value observing(键值监听)与KVB: key-value Binding(键值绑定)

这两个机制是结合起来使用的,分别说下,

Key-Value Observing (简写为KVO):当指定的对象的属性被修改了,允许对象接受到通知的机制。每次指定的被观察对象的属性被修改的时候,KVO都会自动的去通知相应的观察者。

KVO的优点

当 有属性改变,KVO会提供自动的消息通知。这样的架构有很多好处。首先,开发人员不需要自己去实现这样的方案:每次属性改变了就发送消息通知。这是KVO机制提供的最大的优点。因为这个方案已经被明确定义,获得框架级支持,可以方便地采用。开发人员不需要添加任何代码,不需要设计自己的观察者模型,直接可以在工程里使用。其次,KVO的架构非常的强大,可以很容易的支持多个观察者观察同一个属性,以及相关的值。

KVB实现的两个基本方法
1:为对象添加观察者OBserver addObserver:forKeyPath:options:context:
2:观察者OBserver收到信息的处理函数  observeValueForKeyPath:ofObject:change:context:

KVO和KVB最明显的使用场景就是在一些界面实时显示行很强的地方,比如股票走向、售票余额等,这种方式免去了我们自己操作通知的麻烦,想到这,我发现当初点金和91市场中下载页面进度的显示也完全可以采用这种方式。

先说到这儿吧,更精彩的待续。。。。。。。。

原英文链接:http://maniacdev.com/2012/12/helper-classes-for-easy-cocoa-touch-key-value-observing-kvo-and-key-value-binding-kvb/

 

 

1、setValue:forKey: 可以访问到setXXX:方法。

2、valueForKey:可以访问到实例变量和不带参数的方法。

3、performSelector可以访问所有方法。

 


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
目标检测(Object Detection)是计算机视觉领域的一个核心问题,其主要任务是找出图像所有感兴趣的目标(物体),并确定它们的类别和位置。以下是对目标检测的详细阐述: 一、基本概念 目标检测的任务是解决“在哪里?是什么?”的问题,即定位出图像目标的位置并识别出目标的类别。由于各类物体具有不同的外观、形状和姿态,加上成像时光照、遮挡等因素的干扰,目标检测一直是计算机视觉领域最具挑战性的任务之一。 二、核心问题 目标检测涉及以下几个核心问题: 分类问题:判断图像的目标属于哪个类别。 定位问题:确定目标在图像的具体位置。 大小问题:目标可能具有不同的大小。 形状问题:目标可能具有不同的形状。 三、算法分类 基于深度学习的目标检测算法主要分为两大类: Two-stage算法:先进行区域生成(Region Proposal),生成有可能包含待检物体的预选框(Region Proposal),再通过卷积神经网络进行样本分类。常见的Two-stage算法包括R-CNN、Fast R-CNN、Faster R-CNN等。 One-stage算法:不用生成区域提议,直接在网络提取特征来预测物体分类和位置。常见的One-stage算法包括YOLO系列(YOLOv1、YOLOv2、YOLOv3、YOLOv4、YOLOv5等)、SSD和RetinaNet等。 四、算法原理 以YOLO系列为例,YOLO将目标检测视为回归问题,将输入图像一次性划分为多个区域,直接在输出层预测边界框和类别概率。YOLO采用卷积网络来提取特征,使用全连接层来得到预测值。其网络结构通常包含多个卷积层和全连接层,通过卷积层提取图像特征,通过全连接层输出预测结果。 五、应用领域 目标检测技术已经广泛应用于各个领域,为人们的生活带来了极大的便利。以下是一些主要的应用领域: 安全监控:在商场、银行
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值