NSPredicate 比较字符串小于其他字符串

Posted

技术标签:

【中文标题】NSPredicate 比较字符串小于其他字符串【英文标题】:NSPredicate compare string less than other string 【发布时间】:2011-02-17 10:06:12 【问题描述】:

编辑:忽略这个问题。我没有正确查看我的测试数据。我下面的代码有效。 但是有一些很好的答案值得一看。谢谢!

为方便起见,我们将所有数据作为字符串存储在核心数据中。现在我想获取时间戳早于指定时间戳 (oldDate) 的实体的所有对象。这意味着我想与 NSPredicate 进行比较,以便 NSFetchRequest 返回时间戳小于指定时间戳的对象。以下显然不起作用,但这是迄今为止我想出的最好的。

- (void)deleteRecordsOlderThan:(NSString*)oldDate 
    NSError *error = nil;   
    NSFetchRequest *request =[[NSFetchRequest alloc] init];
    [request setEntity:[NSEntityDescription entityForName:tableName inManagedObjectContext:context]];
    [request setIncludesPropertyValues:NO];
    [request setIncludesPendingChanges:YES];
    NSPredicate *predicate = [NSPredicate predicateWithFormat:@"timestamp < %@)",oldDate];
    NSLog(@"Predicate: %@",predicate);
    [request setPredicate:predicate];

    NSArray *filteredObjects = [context executeFetchRequest:request error:&error];
    [request release];
    if (error) 
        NSLog(@"Error fetching objects for deletion: %@", [error description]);
        return;
    
    NSLog(@"Number of old objects to delete:%d",filteredObjects.count);
    for (NSManagedObject * nsObject in filteredObjects) 
        [context deleteObject:nsObject];
    

我希望你明白我的意思。有点难以解释。

【问题讨论】:

【参考方案1】:

如果你可以支持块,那么你可以这样写谓词:

NSPredicate *predicate = [NSPredicate predicateWithBlock:^(id evaluatedObject, NSDictionary *bindings) 
    return ([[NSDate dateWithTimeIntervalSince1970:[[evaluatedObject timestamp] doubleValue] timeIntervalSinceDate:oldDate] < 0);
];

所做的只是从您的字符串中创建一个 NSDate(假设它是自 1970 年以来的时间戳),然后检查自 oldDate 以来的时间间隔是否小于零,表明它在 oldDate 之前。修改不等式以适合您的逻辑。

【讨论】:

我认为这会拯救我,但不幸的是它说明了 4.0 之后可用的内容......而且我们的目标是 3.1.3 及更高版本。我现在就去找一堵好墙来撞我的头。【参考方案2】:

如果时间戳存储为自参考日期以来的秒数,例如 unix 纪元,那么您可以使用:

NSPredicate *predicate = [NSPredicate predicateWithFormat:@"timestamp < %@)", [NSString stringWithFormat:@"%lf", [oldDate timeIntervalSince1970]]];

查看NSDate documentation,这些实例方法返回的 NSTimeInterval 实际上只是一个 double。

【讨论】:

如果时间戳在数据模型中具有 String 类型,这是否有效?如果是日期或数字,我不会有这个问题。 稍微改变了这个答案。我很确定 NSPredicate 足够聪明,可以为 NSStrings 调用正确的 compare: 方法。【参考方案3】:

您可以更改核心数据实体,使其存储日期而不是表示日期的字符串。

这将使这变得非常容易。

【讨论】:

我知道,但由于其他原因,我不能这样做。我不是数据模型的所有者。【参考方案4】:

这将取决于日期的存储方式,例如“2010-02-17”或自 1970 年以来的秒数,但我想添加一个临时属性将是最简单的解决方案(除了更改日期属性,你说你不能这样做)。

瞬态属性是一种数据模型属性,它不保存在数据库中,但每次都会计算。因此,您可以创建一个瞬态 Integer 属性,该属性是根据您的字符串值计算得出的。有关制作此类 getter 函数的信息,请参阅 2pi.dk/tech/cocoa/transient_properties.html 上的“更好的解决方案”。

【讨论】:

时间戳是自 1970 年以来的秒数。我现在觉得有点傻,但是“添加瞬态属性”是什么意思? 时间戳,好吧,这些很容易使用。但是为什么你把它存储为一个字符串而不是一个整数呢?数据模型绝对没有什么可以改变的吗?无论如何,瞬态属性是不保存在数据库中的数据模型属性,而是每次都会计算的。因此,您可以创建一个瞬态 Integer 属性,该属性是根据您的字符串值计算得出的。有关制作此类 getter 函数的信息,请参阅 2pi.dk/tech/cocoa/transient_properties.html 上的“更好的解决方案”。

以上是关于NSPredicate 比较字符串小于其他字符串的主要内容,如果未能解决你的问题,请参考以下文章

在 NSPredicate 中比较整数和字符串

在保存到核心数据之前将字符串与 NSPredicate 进行比较

NSPredicate 谓词

C#字符串比较方法Compare()

常用的正则表达式

对数组中的字符串不区分大小写的NSPredicate?