但是,似乎如果我想使用NSPredicate来查找NSManagedObject,使用“uniqueKey ==%@”或“uniqueKey MATCHES [cd]%@”,它就不能正常工作了.
它总是错过匹配的对象,直到我将匹配对象的uniqueKey的属性更改为具有特定的类,如NSString或NSNumber.
有人可以解释使用NSPredicate和Transformable属性的限制吗?
解决方法
由于这是“核心数据可转换价值搜索nspredicate”(我搜索试图找到灵感)的第一个谷歌热门,我想添加我的工作答案.
如何使用具有可转换属性的NSPredicate
简短,令人兴奋的回答:你需要聪明地对待你的数据变换器.您需要将值转换为NSData,其中包含我称之为“原始标识信息”的内容,即可用于重建对象的最小,最具识别性的字节集.答案很长,……
最重要的是,考虑:
>您实际意味着使用可转换属性吗?如果任何支持的数据类型 – 甚至二进制数据 – 就足够了,请使用它.
>您了解实际上可转换的属性是什么吗?他们如何打包和打开商店的数据?在Apple的文档中查看Non-Standard Persistent Attributes.
>阅读上述内容后,请问:隐藏支持类型“支持属性”的自定义代码是否适用于您?可能使用这种技术.
现在,经过这些考虑,可转换属性相当光滑.坦率地说,为NSoo编写一个NSValueTransformer“FooToData”给NSData似乎比编写大量自组织自定义代码更清晰.我没有找到Core Data不知道需要使用已注册的NSValueTransformer转换数据的情况.
要继续简单地解决这些问题:
>您是否告诉Core Data使用什么变压器?在表视图中打开Core Data模型,单击实体,单击属性,加载Data Model Inspector窗格.在“属性类型:可转换”下,将“名称”设置为变换器.
>使用默认转换器(再次,请参阅以前的Apple文档)或编写自己的转换器 – transformedValue:必须返回NSData.
> NSKeyedUnarchiveFromDataTransformerName是默认的转换器,可能不够,或者可能会绘制一些瞬态实例数据,这些数据可以使两个相似的对象在它们相等时不同.
>转换后的值应仅包含 – 我称之为“原始识别信息”.商店将比较字节,因此每个字节都是重要的.
>您也可以在全球范围内注册变压器.我必须这样做,因为我实际上在应用程序的其他地方重用它们 – 例如NSString * name = @“FooTrans”; [NSValueTransformer setValueTransformer:[NSClassFromString(name)new] forName:name];
您可能不希望使用变换严格查询的数据操作 – 例如主要信息使用变压器的大型导入 – 哎呀!
然后最后,我只是使用它来测试具有NSPredicates的模型上的高级对象属性的相等性 – 例如“%K ==%@” – 它运行正常.我没有尝试过一些不同的匹配术语,但如果他们有时工作我会不会感到惊讶,而其他人则不会.
这是一个NSURL到NSData转换器的示例.为什么不直接存储字符串?是的,没关系 – 这是屏蔽存储属性的自定义代码的一个很好的例子.此示例说明了一个额外的字节被添加到字符串化URL以记录它是否是文件URL – 允许我们知道在解压缩对象时要使用的构造函数.
// URLToDataTransformer.h - interface extern NSString *const kURLToDataTransformerName; @interface URLToDataTransformer : NSValueTransformer @end
…
// URLToDataTransformer.m - implementation #import "URLToDataTransformer.h" NSString *const kURLToDataTransformerName = @"URLToDataTransformer"; @implementation URLToDataTransformer + (Class)transformedValueClass { return [NSData class]; } + (BOOL)allowsReverseTransformation { return YES; } - (id)transformedValue:(id)value { if (![value isKindOfClass:[NSURL class]]) { // Log error ... return nil; } NSMutableData *data; char fileType = 0; if ([value isFileURL]) { fileType = 1; data = [NSMutableData dataWithBytes:&fileType length:1]; [data appendData:[[(NSURL *)value path] dataUsingEncoding:NSUTF8StringEncoding]]; } else { fileType = -1; data = [NSMutableData dataWithBytes:&fileType length:1]; [data appendData:[[(NSURL *)value absoluteString] dataUsingEncoding:NSUTF8StringEncoding]]; } return data; } - (id)reverseTransformedValue:(id)value { if (![value isKindOfClass:[NSData class]]) { // Log error ... return nil; } NSURL *url = nil; NSData *data = (NSData *)value; char fileType = 0; NSRange range = NSMakeRange(1,[data length]-1); [data getBytes:&fileType length:1]; if (1 == fileType) { NSData *actualData = [data subdataWithRange:range]; NSString *str = [[NSString alloc] initWithData:actualData encoding:NSUTF8StringEncoding]; url = [NSURL fileURLWithPath:str]; } else if (-1 == fileType) { NSData *actualData = [data subdataWithRange:range]; NSString *str = [[NSString alloc] initWithData:actualData encoding:NSUTF8StringEncoding]; url = [NSURL URLWithString:str]; } else { // Log error ... return nil; } return url; } @end