4

我们知道 NSFetchRequest 的 propertiesToFetch 可以限制我们想要的属性,它可以减少内存的占用。如果我们这样做:

request.propertiesToFetch = [NSArray arrayWithObjects:@"nID", nil];

和:

    NSString *strID = [NSString stringWithFormat:@"%@", aPerson.nID];
cell.textLabel.text = strID;

然后调试输出(而不是触发故障):

CoreData: sql: SELECT t0.Z_ENT, t0.Z_PK, t0.ZNID FROM ZPERSON t0 ORDER BY t0.ZNID

然而,当我向 NSManagedObject Person 的子类添加一个新的未建模属性时,发生了一件奇怪的事情。像这样:

@interface Person (Ex)
@property (nonatomic, retain) NSString *strTempName;
@end

@implementation Person (Ex)
@dynamic strTempName;

-(void)setStrTempName:(NSString *)strTempName
{
    [self setPrimitiveValue:strTempName forKey:@"strTempName"];
}

-(NSString*)strTempName
{
    return [self primitiveValueForKey:@"strTempName"];
}

这是访问新的未建模属性的地方:

NSString *strTT = aPerson.strTempName;

然后调试输出:

2013-05-06 10:43:07.789 TestCoreData[988:c07] CoreData: sql: SELECT 0, t0.Z_PK, t0.Z_OPT, t0.ZBISFORTEST, t0.ZBISINTRASH, t0.ZNID, t0.ZSTRNAME, t0.ZSTROK, t0.ZSTROK2, t0.ZSTRTEST1, t0.ZSTRTEST2, t0.ZADEPART, t0.ZAMIDDLETHUMNAIL, t0.ZANORMALPIC, t0.ZASMALLTHUMNAIL FROM ZPERSON t0 WHERE  t0.Z_PK = ? 
2013-05-06 10:43:07.790 TestCoreData[988:c07] CoreData: annotation: sql connection fetch time: 0.0010s
2013-05-06 10:43:07.791 TestCoreData[988:c07] CoreData: annotation: total fetch execution time: 0.0018s for 1 rows.
2013-05-06 10:43:07.792 TestCoreData[988:c07] CoreData: annotation: fault fulfilled from database for : 0x898a030 <x-coredata://CAD35D43-F6B2-4463-B59B-C9A3CD488935/Person/p51846>

从上面的输出消息中,我们可以找到未建模的属性,而不是 SELECT t0.Z_ENT, t0.Z_PK, t0.ZNID,生成 SELECT all properties 语句并触发错误!

但是,我已经阅读了一些关于未建模属性的消息(这里是链接):

因为未建模的属性只是自定义 NSManagedObject 子类的属性而不是实体,所以故障对象对它们一无所知。故障对象从数据模型初始化,因此它们响应的所有键都必须在数据模型中。这意味着故障将无法可靠地响应对未建模属性的请求。

为什么会发生奇怪的事情?

提前致谢。


@匿名的,

选择 1):应用程序崩溃并且调试输出显示:

2013-05-06 14:03:05.665 TestCoreData[1794:c07] -[Person strTempName]: unrecognized selector sent to instance 0x6ba77b0
2013-05-06 14:03:30.395 TestCoreData[1794:c07] *** Terminating app due to uncaught exception 'NSInvalidArgumentException', reason: '-[Person strTempName]: unrecognized selector sent to instance 0x6ba77b0'

选择2):首先,willAccessValueForKey等方法是发送KVO更改通知,就像NSManagedObject.h中的注释一样:

- (void)willAccessValueForKey:(NSString *)key;      // read notification
- (void)didAccessValueForKey:(NSString *)key;       // read notification (together with willAccessValueForKey used to maintain inverse relationships, to fire faults, etc.) 

其次,在选择 2) 代码之后,奇怪的事情(触发故障)在我的应用程序中仍然存在。

也感谢您的回复。

4

3 回答 3

0

有一些事情出错了:

首先:这是正确的,@dynamic 是错误的。访问器在您的代码中。没有什么动态的。

第二:原始访问器用于建模属性。你有一个“ivar 属性”。只需像在其他任何地方使用它一样使用它。(但添加 ...access... 和 ...change... 方法。)

更改后,重新检查是否触发了故障。

+++ 在 -didAccessValueForKey: (NSManagedObject) 的文档中,您将找到正确实现的示例:

- (NSString *)firstName
{
    [self willAccessValueForKey:@"firstName"];
    NSString *rtn = firstName;
    [self didAccessValueForKey:@"firstName"];
    return rtn;
}
于 2013-05-06T21:24:45.210 回答
0

return [self primitiveValueForKey:@"strTempName"];

这是错误的。primitiveValueForKey用于访问建模属性。由于您的属性未建模,因此您会遇到所有奇怪的行为 - 核心数据会触发错误,因为它正在寻找该属性,而当它没有找到它时,它会引发异常。

你想达到什么目标?也许您正在寻找一个瞬态属性,或者您应该像在任何其他对象中一样合成该属性?

于 2013-05-06T21:35:04.543 回答
-1

我认为你编码太多了。@dynamic 为您生成 getter 和 setter。你有两个选择。

1) 删除 setter/getter 调用。

2) 修改您的 getter/setter 以警告系统您正在更改值:

-(void)setStrTempName:(NSString *)strTempName
{
    [self willChangeValueForKey:@"strTempName"];
    [self setPrimitiveValue:strTempName forKey:@"strTempName"];
    [self didChangeValueForKey:@"strTempName"];
}

-(NSString*)strTempName
{
    [self willAccessValueForKey:@"strTempName"];
    NSString *tmp = [self primitiveValueForKey:@"strTempName"];
    [self didAccessValueForKey:@"strTempName"];
    return tmp;
}
于 2013-05-06T04:26:33.153 回答