-setPrimitiveValue:forKey:不会触发KVO通知。但在我的大脑中,KVO只有在事情发生变化时才有意义。但是,当我只对其进行读访问时,如何更改它呢?
-primitiveValueForKey:只获取某个键的对象。但它不会修改它。那么,在使用-valueForKey:时,这为什么会/是否会导致KVO通知呢
(当然,这是有道理的,但我还看不出来。)
发布于 2010-11-18 07:50:57
核心数据主要使用-primitiveValueForKey:和-setPrimitiveValue:forKey:方法。特别是,Core Data不只需要知道您何时要修改属性;它还需要知道您何时要访问它,这样它就可以实现出错。
因此,核心数据添加了在getter中使用的-{will,did}AccessValueForKey:方法,就像-{will,did}ChangeValueForKey:方法在setter中用作KVO钩子一样。
然而,还有另一个问题:核心数据实际上也为您管理建模属性的底层存储。因此,您需要一些方法来在-{will,did}{Access,Change}ValueForKey:方法建立的屏障内操作这种底层存储。这就是-primitiveValueForKey:和-setPrimitiveValue:forKey:的用武之地。
这就是为什么在@property和@dynamic出现之前,实现核心数据getter和setter的标准模式是这样的:
// Person.m
#import "Person.h"
@implementation Person
- (NSString *)name {
[self willAccessValueForKey:@"name"];
NSString *value = [self primitiveValueForKey:@"name"];
[self didAccessValueForKey:@"name"];
}
- (void)setName:(NSString *)value {
[self willChangeValueForKey:@"name"];
[self setPrimitiveValue:value forKey:@"name"];
[self didChangeValueForKey:@"name"];
}
@end当然,现在,如果您希望核心数据在运行时为您生成以下内容,您可以只声明一个属性并将其定义为@dynamic:
// Person.h
@interface Person : NSManagedObject
@property (nonatomic, readwrite, copy) NSString *name;
@end
// Person.m
#import "Person.h"
@implementation Person
@dynamic name;
@end但是,在某些情况下,您仍然希望在没有 KVO或故障触发的情况下操作底层存储。因此,Core Data也提供了一种新的方法来实现这一点,也是围绕属性声明和自动合成构建的:
// Person.h
@interface Person : NSManagedObject
@property (nonatomic, readwrite, copy) NSString *name;
@end
// Person.m
#import "Person.h"
@interface Person ()
@property (nonatomic, readwrite, copy) NSString *primitiveName;
@end
@implementation Person
@dynamic name;
@dynamic primitiveName;
@end请注意,我将类continuation放在了.m文件中;Person外部的代码(甚至-awakeFromInsert和-awakeFromFetch外部的代码)都不应该接触到它。但它确实让我获得了"name“属性的底层存储,而不是在我的代码中嵌入文字字符串,而是使用了真正的类型。
https://stackoverflow.com/questions/4147020
复制相似问题