注意: 我不确定自2011年5月(来自Scott Ahten的已接受答案)以来是否发生了更改,但您绝对可以在可转换属性上使用NSPredicate进行搜索。Scott正确地解释了为什么您的假设是错误的,但如果“有人能解释使用Transformable属性时NSPredicate的限制吗?”是您的问题,他暗示这是不可能的,这是不正确的。
由于这是谷歌搜索“Core Data transformable value search nspredicate”的第一个结果(我试图寻找灵感),我想添加我的工作答案。
如何使用可转换属性的NSPredicate
简短而精辟的答案:您需要聪明地使用数据转换器。您需要将值转换为包含我称之为“原始标识信息”的NSData,即最小的、最具标识性的字节集,可用于重构您的对象。长答案...
首先,请考虑以下内容:
- 您是否真的要使用可转换属性?如果任何支持的数据类型-甚至是二进制数据-都可以满足要求,请使用它。
- 您是否了解可转换属性实际上是什么?它们如何将数据打包和解包到存储中?请查看Apple文档中的非标准持久属性。
- 阅读上述内容后,请问:隐藏支持的类型“后备属性”的自定义代码是否适用于您?可能会使用该技术。
现在,在考虑这些因素之后,可转换属性相当不错。坦白地说,为Foo实例编写一个NSValueTransformer“FooToData”,将其转换为NSData,似乎比编写大量的特定的自定义代码更清晰。我还没有找到Core Data不知道需要使用注册的NSValueTransformer转换数据的情况。
要继续,请简单地解决以下问题:
- 你是否告诉了Core Data要使用哪个转换器?在表格视图中打开Core Data模型,单击实体,单击属性,加载数据模型检查器面板。在“属性类型:可转换”下,将“名称”设置为您的转换器。
- 使用默认转换器(再次参见之前的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
...
#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]])
{
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]])
{
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
{
return nil;
}
return url;
}
@end