利用KVC和associative特性在NSObject中存储键值_移动开发_编程开发_程序员俱乐部

中国优秀的程序员网站程序员频道CXYCLUB技术地图
热搜:
更多>>
 
您所在的位置: 程序员俱乐部 > 编程开发 > 移动开发 > 利用KVC和associative特性在NSObject中存储键值

利用KVC和associative特性在NSObject中存储键值

 2013/9/16 14:46:27  PowerAuras  博客园  我要评论(0)
  • 摘要:KVC一直没仔细看过KVC的用法,想当然的认为可以在NSObject对象中存入任意键值对,结果使用时碰到问题了。一个简单的位移动画:CAKeyframeAnimation*keyPosi=[CAKeyframeAnimationanimationWithKeyPath:@"position"];keyPosi.path=path.CGPath;keyPosi.delegate=self;[label.layeraddAnimation:keyPosiforKey:@"x"]
  • 标签:利用 SSO

KVC 

 一直没仔细看过KVC的用法,想当然的认为可以在NSObject对象中存入任意键值对,结果使用时碰到问题了。

 

  一个简单的位移动画:

CAKeyframeAnimation *keyPosi=[CAKeyframeAnimation animationWithKeyPath:@"position"];
keyPosi.path=path.CGPath;
keyPosi.delegate=self;
[label.layer addAnimation:keyPosi forKey:@"x"];

  我想要在动画结束后把UILabel从屏幕上移除,于是加上动画结束后的回调:

- (void)animationDidStop:(CAAnimation *)anim finished:(BOOL)flag{

}

  然而看上去从回调方法的参数中似乎无法得到UILabel对象(也许提供的有api可以获得UIlabel,希望各位看官不吝赐教),如果只是为了在这里得到UILabel对象而去设置一个全局变量指针指向它感觉没必要,这时候我想起了KVC,于是在创建动画的时候加上一句:

[keyPosi setValue:label forKey:@"xx"];

  这样在动画结束的回调方法中可以通过anim参数获得UILabel对象了:

class="p1">- (void)animationDidStop:(CAAnimation *)anim finished:(BOOL)flag{

    if (flag) {

        UIView *vie=[anim valueForKey:@"xx"];

        [vie removeFromSuperview];

    }

}

 

   效果已经实现,然而KVC是这样使用的吗?再看一个例子

NSObject *obj = [[NSObject alloc] init];
[obj setValue:@"asd"  forKey:@"xx"];

  运行报错:

*** Terminating app due to uncaught exception 'NSUnknownKeyException', reason: '[<NSObject 0xf65f090> setValue:forUndefinedKey:]: this class is not key value coding-compliant for the key xx.'

  原来setValue:forKey、setValue:forKeyPath的key(Path)参数不是随便写的,必须是类中定义的成员变量名(或者实例方法),这篇文章写的很清楚:http://www.cnblogs.com/jay-dong/archive/2012/12/13/2815778.html。话说回来,为啥前面CAKeyframeAnimation使用的setValue:forKey的key可以成功设置?类里肯定没有xx这个成员变量。继续查资料发现有个方法setValue:forUndefinedKey:,

setValue:forUndefinedKey:
Invoked by setValue:forKey: when it finds no property for a given key.
- (void)setValue:(id)value forUndefinedKey:(NSString *)key Discussion Subclasses can override this method to handle the request in some other way. The default implementation raises an NSUndefinedKeyException.

  当setValue方法的key参数在类中找不到对应成员时,会调用这个方法,重写它可以阻止抛出NSUndefinedKeyException异常

  这样看来,CAKeyFrameAnimation类(或者它的父类)应该是重写了setValue:forUndefinedKey:,然而方法里是怎样处理从而使得valueForKey可以正确取到值呢?

associative

  objective-c的扩展机制有两个特性:category和associative。category扩展类别,associative扩展属性。使用associative需要导入<objc/runtime.h>头文件

  利用category扩展NSObject类别,利用associative和setValue:forUndefinedKey:让NSObject能够存入任意键值对。

 

#import <objc/runtime.h>
@interface NSObject (KVC)
- (id)valueForUndefinedKey:(NSString *)key;
- (void)setValue:(id)value forUndefinedKey:(NSString *)key;
@end


@implementation NSObject(KVC)
- (id)valueForUndefinedKey:(NSString *)key{
    return objc_getAssociatedObject(self, key);
}
- (void)setValue:(id)value forUndefinedKey:(NSString *)key{
    if ([value isKindOfClass:[NSString class]]) {
        objc_setAssociatedObject(self, key, value, OBJC_ASSOCIATION_COPY_NONATOMIC);
    }else{
        objc_setAssociatedObject(self, key, value, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
    }

}
@end

 

  现在再试下:

NSObject *obj = [[NSObject alloc] init];
[obj setValue:@"asd"  forKeyPath:@"xx"];
NSLog(@"%@",[obj valueForKey:@"xx"]);

  可以输出了吧。

 

发表评论
用户名: 匿名