5

对于我的一生,我似乎无法让它发挥作用。

假设我们的实体是一个具有状态字段和订单字段的托管对象。

我将如何获取具有多个相同订单的所有orderedEntries?

请没有答案告诉我只在主谓词中使用 @count 进行子查询,因为我知道该解决方案,所以这篇文章的重点是了解如何在核心数据中使用 having 谓词,这可能比无论如何,子查询。(除非你解释为什么我不能使用having子句)

以下代码将返回一个字典数组,其中包含每个订单号的订单数。我想要的是能够添加一个having子句来限制我的请求只返回代表那些计数大于1的订单对象的字典。

这是到目前为止的代码和我对谓词的尝试:

NSFetchRequest *fetchRequest = [[NSFetchRequest alloc] init];
NSEntityDescription *entity = [NSEntityDescription entityForName:@"OrderedEntry"
                                          inManagedObjectContext:context];
[fetchRequest setEntity:entity];
[fetchRequest setResultType:NSDictionaryResultType];

NSPredicate *predicate = [NSPredicate predicateWithFormat:
                 @"(status == %@)",[NSNumber numberWithInt:EntryStatusAlive]];


[fetchRequest setPredicate:predicate];


NSExpression *keyPathExpression = [NSExpression expressionForKeyPath: @"order"]; // Does not really matter
NSExpression *maxExpression = [NSExpression expressionForFunction: @"count:"
                                                        arguments: [NSArray arrayWithObject:keyPathExpression]];
NSExpressionDescription *expressionDescription = [[NSExpressionDescription alloc] init];
[expressionDescription setName: @"orderCount"];
[expressionDescription setExpression: maxExpression]; 
[expressionDescription setExpressionResultType: NSInteger32AttributeType];

[fetchRequest setPropertiesToFetch: [NSArray arrayWithObjects:expressionDescription,@"order",nil]];


[fetchRequest setPropertiesToGroupBy:[NSArray arrayWithObjects:@"order",nil]];
//[fetchRequest setHavingPredicate:[NSPredicate predicateWithFormat:@"@count > 1"]];
//[fetchRequest setHavingPredicate:[NSComparisonPredicate predicateWithLeftExpression:maxExpression rightExpression:[NSExpression expressionForConstantValue:[NSNumber numberWithInteger:1]] modifier:NSDirectPredicateModifier type:NSGreaterThanPredicateOperatorType options:NSCaseInsensitivePredicateOption]];

NSError *error;

NSArray * array = [context executeFetchRequest:fetchRequest error:&error];
4

3 回答 3

1

我最终为任何感兴趣的人选择了这个

-(BOOL)ordersAreSaneOnDay:(NSNumber*)dayNumber forUser:(User*)user inContext:(NSManagedObjectContext*)context {
NSFetchRequest *fetchRequest = [[NSFetchRequest alloc] init];
NSEntityDescription *entity = [NSEntityDescription entityForName:@"BasicEntry"
                                          inManagedObjectContext:context];
[fetchRequest setEntity:entity];
[fetchRequest setResultType:NSDictionaryResultType];

NSPredicate *predicate = [NSPredicate predicateWithFormat:
                 @"(status == %@) && ((type != %@) && (type != %@) && (dayNumber == %@))  && ((user == NIL) || (user == %@))",[NSNumber numberWithInt:EntryStatusAlive],[NSNumber numberWithInt:EntryTypeTask],[NSNumber numberWithInt:EntryTypeCompletedTask],dayNumber,user];


[fetchRequest setPredicate:predicate];


NSExpression *keyPathExpression = [NSExpression expressionForKeyPath: @"order"]; // Does not really matter
NSExpression *maxExpression = [NSExpression expressionForFunction: @"count:"
                                                        arguments: [NSArray arrayWithObject:keyPathExpression]];
NSExpressionDescription *expressionDescription = [[NSExpressionDescription alloc] init];
[expressionDescription setName: @"orderCount"];
[expressionDescription setExpression: maxExpression]; 
[expressionDescription setExpressionResultType: NSInteger32AttributeType];

[fetchRequest setPropertiesToFetch: [NSArray arrayWithObjects:expressionDescription,@"order",nil]];
[expressionDescription release];

[fetchRequest setPropertiesToGroupBy:[NSArray arrayWithObjects:@"order",nil]];
//[fetchRequest setHavingPredicate:[NSPredicate predicateWithFormat:@"self.order.@count > 1"]];
//[fetchRequest setHavingPredicate:[NSComparisonPredicate predicateWithLeftExpression:maxExpression rightExpression:[NSExpression expressionForConstantValue:[NSNumber numberWithInteger:1]] modifier:NSDirectPredicateModifier type:NSGreaterThanPredicateOperatorType options:NSCaseInsensitivePredicateOption]];

NSError *error;

NSArray * array = [context executeFetchRequest:fetchRequest error:&error];
array = [array filteredArrayUsingPredicate:[NSPredicate predicateWithFormat:@"orderCount > 1"]];
//NSLog(@"it worked %@",array);
[fetchRequest release];

if ([array count]) return FALSE;
return TRUE;

}

于 2013-01-22T15:08:19.690 回答
1

我已经使用以下方法进行了这项工作:

[fetchRequest setHavingPredicate:[NSPredicate predicateWithFormat:@"$orderCount > 1"]];

使用您的 expressionDecription 的名称作为变量$orderCount。或者,您可以使用

NSExpression *countExpression = [NSExpression expressionForVariable:@"orderCount"];
[fetchRequest setHavingPredicate:[NSPredicate predicateWithFormat:@"%@ > 1", countExpression]];
于 2018-06-22T09:50:25.690 回答
0

首先,每当我尝试与您正在做的事情类似的事情时,我都会收到一个错误,您无法将一对多关系传递给setPropertiesToFetch:. NSFetchRequest 文档支持这一点:“属性描述可能表示属性、一对一关系或表达式。 ”这就是问题 #1。

问题 #2 是您似乎也不能按对多关系进行分组(这在文档中没有明确说明,但您会遇到相同的错误,这也是有道理的)。

从要获取的属性中删除“订单”。按属性分组。修改您的主要谓词以仅包含您分组的那些属性(或仅将其删除)。在您的谓词中指定“顺序”。

[fetchRequest setPropertiesToGroupBy: @[ @"???" ]];
[fetchRequest setHavingPredicate: [NSPredicate predicateWithFormat: @"self.order.@count > 1"]];

现在您会看到请求会起作用,但结果可能不是您所期望的:

 - NSArray
     - NSDictionary { status = @"alive", orderCount = "4" }
     - NSDictionary { status = @"alive", orderCount = "9" }
     - NSDictionary { status = @"alive", orderCount = "2" }
     - etc...

NSDictionaryResultType实际上并没有给你任何东西来识别这些对象——它只是给你值。

因此,下一步是取回这些OrderedEntry对象的 ID。诀窍是包含一个表达式,它将返回 NSManagedObjectID 作为每个字典中的键。

我不知道这是否真的会给您带来更好的性能(不仅仅是将其与主要谓词相结合)。根据我的经验,提高获取性能的最佳方法之一是创建单例 NSPredicates 并使用替换变量来设置每次获取。谓词解析可能很昂贵。

字典结果类型可能很痛苦。通常只构造一个好的谓词就更好了。我倾向于只将它们用于表达式(即在返回值的图形上执行统计类型的计算)。如果您查看有关要获取和分组的属性的所有限制以及谓词限制,这似乎就是 Apple 的意图。我什至不确定是否可以通过分组和使用 having 谓词来做你想做的事情 - 例如,你打算按什么分组?如果按状态(您需要按状态分组以包含在您的谓词中),那不会折叠所有 OrderEntries 并且您不会获得单独的 objectID 和 orderCounts 吗?最好坚持使用主要谓词,而不是分组和拥有谓词。

于 2013-01-21T18:10:55.120 回答