2012-02-06 10 views
7

現在使用中コアデータ。その結果、このタイプのを生成するためにコアデータによる個別カウント、NSFetchedResultsControllerへのNSExpression

SELECT item, COUNT(*) FROM myTable GROUP BY item; 

:私はこれらの線に沿って情報を取得しようとしていた中で一つのテーブルを持っている

+---------+------+-------+ 
| item  | COUNT(*) | 
+---------+------+-------+ 
| group 1  | 2  | 
| group 2  | 5  | 
| group 3  | 8  | 
+---------+------+-------+ 

私はNSExpressionを使用するように明るいアイデアを持っていたコアデータが私のためにすべての仕事を行うことを期待して。私は私の車輪を回転させ始めています。 件数:表現機能が私にクラッシュしています。例外はあまり明確ではありません。 の合計:などの他の式関数を使用しても、アプリケーションはクラッシュしません。

結果をNSFetchedResultsControllerに保存するとよいでしょう。私は他の選択肢を検討しましたが、どれも魅力的ではありません。この場合、コア・データをSQLラッパーとして使用するのではなく、SQL照会を作成して処理する方が理にかなっていますか?

参考ソースコードは以下のとおりです。

NSExpression *keyPathExpression = [NSExpression expressionForKeyPath:@"item"]; 
NSExpression *totalExpression = [NSExpression expressionForFunction:@"count:" arguments:[NSArray arrayWithObject:keyPathExpression]]; 

NSExpressionDescription * totalExpressionDescription = [[NSExpressionDescription alloc] init]; 
[totalExpressionDescription setExpression:totalExpression]; 
[totalExpressionDescription setExpressionResultType:NSInteger64AttributeType]; 
[totalExpressionDescription setName:@"totalInstances"]; 

NSFetchRequest *fetchRequest = [[NSFetchRequest alloc] init]; 
NSEntityDescription *entity = [NSEntityDescription entityForName:@"myEntity" inManagedObjectContext:self.managedObjectContext]; 
NSArray *propertiesToFetch = [[NSArray alloc] initWithObjects:strFieldName, totalExpressionDescription, nil]; 

[fetchRequest setEntity:entity]; 
[fetchRequest setReturnsDistinctResults:YES]; 
[fetchRequest setResultType:NSDictionaryResultType]; 
[fetchRequest setPropertiesToFetch:propertiesToFetch]; 
[fetchRequest setFetchBatchSize:20]; 

NSSortDescriptor *sortDescriptor = [[NSSortDescriptor alloc] initWithKey:strFieldName ascending:YES]; 
NSArray *sortDescriptors = [[NSArray alloc] initWithObjects:sortDescriptor, nil]; 
[fetchRequest setSortDescriptors:sortDescriptors]; 

NSFetchedResultsController* aFetchedResultsController = [[NSFetchedResultsController alloc] initWithFetchRequest:fetchRequest managedObjectContext:[self managedObjectContext] sectionNameKeyPath:nil cacheName:nil]; 

... 

NSError *error = nil; 
if (![self.fetchedResultsController performFetch:&error]) 
{ 
    NSLog(@"Unresolved error %@, %@", error, [error userInfo]); 
    abort(); 
} 

答えて

14

私の不自然な例では、私は私が取得したい「EMAILADDRESS」と呼ばれるプロパティを持つ「人」と呼ばれるエンティティを持っている:私はこの素晴らしいチュートリアルを追いましたカウント。

NSPropertyDescription *propDesc = [[[[model entitiesByName] objectForKey:@"Person"] propertiesByName] objectForKey:@"emailAddress"]; 
NSExpression *emailExpr = [NSExpression expressionForKeyPath:@"emailAddress"]; 
NSExpression *countExpr = [NSExpression expressionForFunction:@"count:" arguments:[NSArray arrayWithObject:emailExpr]]; 
NSExpressionDescription *exprDesc = [[NSExpressionDescription alloc] init]; 
[exprDesc setExpression:countExpr]; 
[exprDesc setExpressionResultType:NSInteger64AttributeType]; 
[exprDesc setName:@"count"]; 

NSFetchRequest *fr = [NSFetchRequest fetchRequestWithEntityName:@"Person"]; 
[fr setPropertiesToGroupBy:[NSArray arrayWithObject:propDesc]]; 
[fr setPropertiesToFetch:[NSArray arrayWithObjects:propDesc, exprDesc, nil]]; 
[fr setResultType:NSDictionaryResultType]; 
NSArray *results = [moc executeFetchRequest:fr error:&error]; 

は、私は1000年の記録をデータベースに事前に移入するコードのスニペットを書いた:

NSArray *emailAddresses = [NSArray arrayWithObjects:@"[email protected]", @"[email protected]", @"[email protected]", @"[email protected]", @"[email protected]", nil]; 
    for (int i = 0; i < 1000; i++) { 
     NSManagedObject *person = [NSEntityDescription insertNewObjectForEntityForName:@"Person" inManagedObjectContext:moc]; 
     [person setValue:[NSNumber numberWithInt:i] forKey:@"aNumber"]; 
     [person setValue:[[NSUUID UUID] UUIDString] forKey:@"aUUID"]; 
     [person setValue:[emailAddresses objectAtIndex:(i % [emailAddresses count])] forKey:@"emailAddress"]; 
    } 

、ここでの結果はそれぞれのメールアドレスに200回を挿入します上記のコードです:

2012-05-31 15:17:42.160 Scratch[16084:10d03] CoreData: sql: SELECT t0.ZEMAILADDRESS, COUNT(t0.ZEMAILADDRESS) FROM ZPERSON t0 GROUP BY t0.ZEMAILADDRESS 
2012-05-31 15:17:42.162 Scratch[16084:10d03] CoreData: annotation: sql connection fetch time: 0.0024s 
2012-05-31 15:17:42.163 Scratch[16084:10d03] CoreData: annotation: total fetch execution time: 0.0029s for 5 rows. 
(gdb) po results 
(NSArray *) $2 = 0x0f811280 <_PFArray 0xf811280>(
{ 
    count = 200; 
    emailAddress = "[email protected]"; 
}, 
{ 
    count = 200; 
    emailAddress = "[email protected]"; 
}, 
{ 
    count = 200; 
    emailAddress = "[email protected]"; 
}, 
{ 
    count = 200; 
    emailAddress = "[email protected]"; 
}, 
{ 
    count = 200; 
    emailAddress = "[email protected]"; 
} 
) 
+1

NSPropertyDescriptionが不足していて、NSFetchRequestが「項目ごとのグループ化」行を追加してSQLクエリとして正しく解釈されている可能性があります。私は投稿時にそのオブジェクトを認識していませんでした。私は2月に答えを返すことができた...笑。これは、私が将来このソリューションを使用することを決めた場合には、知っておいてよかったです。あなたの応答は高く評価されます。 :-) –

+0

1つの注釈は、元の著者がNSFetchedResultsControllerをグループ化して使用すると述べました。残念ながら、 'setPropertiesToGroupBy'が機能するためには、' setResultType:NSDictionaryResultType'が必要です。これは、変更の追跡を無効にします。 http://stackoverflow.com/a/4361198/287403 –

+0

を参照してください。カウントをすべて並べ替えることは可能ですか? – Andy

関連する問題