[英]Core Data Transformable attributes NOT working with NSPredicate
我經常將Transformable
用於Core Data attributes
,因此我可以稍后更改它們。
但是,看起來,如果我想使用NSPredicate
來查找NSManagedObject
,使用"uniqueKey == %@"
或"uniqueKey MATCHES[cd] %@"
,它就不能正常工作。
它總是錯過匹配的對象,直到我將匹配的 object 的 uniqueKey 的屬性更改為具有特定的 class (如NSString
或NSNumber
)。
有人可以解釋使用帶有Transformable
屬性的NSPredicate
的限制嗎?
注意:我不確定自 5/2011 以來何時/是否發生了變化(來自 Scott Ahten 接受的答案),但您絕對可以使用 NSPredicate 搜索可轉換屬性。 斯科特正確地解釋了為什么你的假設被打破了,但是如果有人可以解釋使用帶有可轉換屬性的 NSPredicate 的限制嗎? 是你的問題,他暗示這是不可能的,這是不正確的。
由於這是“ Core Data transformable value search nspredicate ”(我搜索的內容試圖尋找靈感)的第一個谷歌熱門,我想添加我的工作答案。
如何使用具有可轉換屬性的 NSPredicate
簡短而令人興奮的答案:您需要對數據轉換器保持聰明。 您需要將值轉換為包含我稱之為“原始識別信息”的 NSData,即可用於重建 object 的最小、最具識別性的字節集。 長答案,...
首先,考慮:
現在,拋開這些考慮,可轉換的屬性相當巧妙。 坦率地說,為 Foo 實例編寫一個 NSValueTransformer "FooToData" 到 NSData 似乎比編寫大量臨時自定義代碼更干凈。 我還沒有發現 Core Data 不知道它需要使用注冊的 NSValueTransformer 轉換數據的情況。
繼續簡單地解決這些問題:
NSString *name = @"FooTrans"; [NSValueTransformer setValueTransformer:[NSClassFromString(name) new] forName:name];
NSString *name = @"FooTrans"; [NSValueTransformer setValueTransformer:[NSClassFromString(name) new] forName:name];
您可能不想使用轉換大量查詢的數據操作 - 例如,主鍵信息使用轉換器的大型導入 - 哎呀!
最后,我只是使用它來測試具有 NSPredicates 的模型上高級 object 屬性的相等性——例如“%K == %@”——它工作正常。 我還沒有嘗試過一些不同的匹配術語,但如果它們有時有效,而另一些則無效,我不會感到驚訝。
這是 NSURL 到 NSData 轉換器的示例。 為什么不只存儲字符串? 是的,這很好——這是自定義代碼屏蔽存儲屬性的一個很好的例子。 這個例子說明了一個額外的字節被添加到字符串化的 URL 來記錄它是否是一個文件 URL 或者不是 - 讓我們知道當 ZA8CFDE6331BD59EB66696F89111 解包時使用什么構造函數
// 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
可轉換屬性通常作為存檔的二進制數據保存。 因此,您試圖將 NSData 的實例與 NSString 或 NSNumber 的實例進行比較。
由於這些類以不同的方式解釋相同的數據,因此它們不被視為匹配項。
你可以試試這種方式
NSExpression *exprPath = [NSExpression expressionForKeyPath:@"transformable_field"];
NSExpression *exprKeyword = [NSExpression expressionForConstantValue:nsdataValue];
NSPredicate *predicate = [NSComparisonPredicate predicateWithLeftExpression:exprPath rightExpression:exprKeyword modifier:NSDirectPredicateModifier type:NSEqualToPredicateOperatorType options:0];
聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.