我经常将Transformable
用于Core Data attributes
,因此我可以稍后更改它们。
但是,如果我想使用NSPredicate
来查找NSManagedObject
,使用"uniqueKey == %@"
或"uniqueKey MATCHES[cd] %@"
,它似乎无法正常工作。< / p>
它总是错过匹配的对象,直到我将匹配对象的uniqueKey的属性更改为具有NSString
或NSNumber
等特定类。
有人可以解释使用NSPredicate
Transformable
属性的限制吗?
答案 0 :(得分:5)
注意:我不确定自2011年5月以来这是否发生了变化(来自Scott Ahten的接受答案),但您绝对可以使用NSPredicate搜索可转换属性。 Scott正确地解释了为什么你的假设被打破了,但是如果有人可以解释使用NSPredicate和Transformable属性的限制吗?是你的问题,他暗示这是不可能的,那是不正确的。
由于这是“核心数据可转换价值搜索nspredicate ”的第一次谷歌搜索(我搜索试图找到灵感),我想添加我的工作答案。
如何将NSPredicate与可转换属性一起使用
简短而令人兴奋的回答:您需要对数据变换器保持警惕。您需要将值转换为NSData,其中包含我称之为“原始标识信息”的内容,即可用于重建对象的最小,最具识别性的字节集。答案很长,......
最重要的是,考虑一下:
现在,经过这些考虑,可转换属性相当光滑。坦率地说,为NSoo编写一个NSValueTransformer“FooToData”给NSData看起来比编写大量的自定义代码更简洁。我没有发现Core Data不知道需要使用已注册的NSValueTransformer转换数据的情况。
继续简单地解决这些问题:
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
答案 1 :(得分:4)
可转换属性通常作为归档二进制数据保留。因此,您尝试将NSData的实例与NSString或NSNumber的实例进行比较。
由于这些类以不同的方式解释相同的数据,因此它们不被视为匹配。
答案 2 :(得分:-1)
你可以这样试试
NSExpression *exprPath = [NSExpression expressionForKeyPath:@"transformable_field"];
NSExpression *exprKeyword = [NSExpression expressionForConstantValue:nsdataValue];
NSPredicate *predicate = [NSComparisonPredicate predicateWithLeftExpression:exprPath rightExpression:exprKeyword modifier:NSDirectPredicateModifier type:NSEqualToPredicateOperatorType options:0];