Core Data中的Transformable属性无法与NSPredicate一起使用

10
我经常使用Transformable来处理Core Data属性,这样我以后可以更改它们。
然而,如果我想使用NSPredicate查找NSManagedObject,并使用"uniqueKey == %@""uniqueKey MATCHES[cd] %@",它似乎无法正常工作。
直到我将匹配对象的uniqueKey属性更改为特定类别,例如NSStringNSNumber,才会始终缺少匹配的对象。
有人能解释一下在使用Transformable属性时使用NSPredicate的限制吗?

我测试 uniqueKey == %@ 是可以的,但是 MATCHES、contains、like 都不成功。 - user501836
3个回答

7

注意: 我不确定自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

...

// 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

4
您有没有查询该类时实际谓词的示例? - pbuchheit

4

可变属性通常作为归档的二进制数据进行持久化。因此,您正在尝试将NSData实例与NSString或NSNumber实例进行比较。

由于这些类以不同的方式解释相同的数据,它们不被视为匹配。


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

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接