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

问题描述:

编辑:忽略此问题。我没有正确地查看我的测试数据。我的代码如下。 但是有一些很好的答案。谢谢!NSPredicate比较字符串小于其他字符串

为方便起见,我们将所有数据作为字符串存储在核心数据中。现在我想获取具有比指定时间戳(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]; 
    } 
} 

}

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

如果你能支撑块,那么你可以写这样的断言:

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

所有这一切确实是使一个NSDate你的字符串(假设这是自1970年以来时间戳),然后检查查看自oldDate以来的时间间隔是否小于零,表明它在oldDate之前。修改不平等来适应你的逻辑。

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

这将使这非常容易。

+0

我知道,但由于其他原因,我不能这样做。我不是数据模型的所有者。 – sanna 2011-02-17 10:15:46

如果因为参考日期时间戳存储为秒,如Unix纪元,那么你可以使用:

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

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

+0

如果时间戳在数据模型中具有String类型,这是否工作?如果是日期或数字,我不会有这个问题。 – sanna 2011-02-17 11:09:04

+0

稍微更改此答案。我很确定NSPredicate足够聪明,可以为NSStrings调用正确的compare:方法。 – 2011-02-17 12:11:58

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

瞬态属性是一种数据模型属性,它不保存在数据库中,而是每次计算。所以你可以创建一个瞬时的Integer属性,它是从你的字符串值中计算出来的。有关制作这种吸气功能的信息,请参阅2pi.dk/tech/cocoa/transient_properties.html上的“更好的解决方案”。

+0

时间戳是自1970年以来的秒数。现在我觉得有点愚蠢,但是“添加瞬态属性”是什么意思? – sanna 2011-02-17 11:11:44