核心数据可转换属性不适用于NSPredicate

时间:2011-05-24 00:31:28

标签: xcode ios core-data nspredicate transformable

我经常将Transformable用于Core Data attributes,因此我可以稍后更改它们。

但是,如果我想使用NSPredicate来查找NSManagedObject,使用"uniqueKey == %@""uniqueKey MATCHES[cd] %@",它似乎无法正常工作。< / p>

它总是错过匹配的对象,直到我将匹配对象的uniqueKey的属性更改为具有NSStringNSNumber等特定类。

有人可以解释使用NSPredicate Transformable属性的限制吗?

3 个答案:

答案 0 :(得分:5)

注意:我不确定自2011年5月以来这是否发生了变化(来自Scott Ahten的接受答案),但您绝对可以使用NSPredicate搜索可转换属性。 Scott正确地解释了为什么你的假设被打破了,但是如果有人可以解释使用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

答案 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];