Различный счетчик через основные данные, выражение NSExpression в NSFetchedResultsController - PullRequest
7 голосов
/ 06 февраля 2012

В настоящее время используется Базовые данные . У меня есть одна таблица, в которой я пытаюсь получить информацию следующим образом:

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

для получения результата такого типа:

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

У меня появилась блестящая идея использовать NSExpression , в надежде на то, что Core Data сделает всю работу за меня. Я начинаю крутить колеса. Функция выражения : не работает. Исключение не очень понятно. Использование других функций выражений, таких как sum: , не приводит к сбою приложения.

Было бы неплохо сохранить результаты в NSFetchedResultsController . Я изучил другие варианты, ни один из которых не слишком привлекателен. Будет ли разумнее написать SQL-запрос и покончить с ним, а не использовать Core Data в качестве оболочки 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();
}

Ответы [ 2 ]

14 голосов
/ 01 июня 2012

Это возможно с NSExpression, как вы изначально предлагали, в моем надуманном примере у меня есть сущность под названием «Person» со свойством «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:@"test@test.com", @"1@1.com", @"1@2.com", @"roam@test.com", @"foo@foo.com", 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 = "1@1.com";
},
{
    count = 200;
    emailAddress = "1@2.com";
},
{
    count = 200;
    emailAddress = "foo@foo.com";
},
{
    count = 200;
    emailAddress = "roam@test.com";
},
{
    count = 200;
    emailAddress = "test@test.com";
}
)
0 голосов
/ 08 февраля 2012

Было больше смысла просто написать собственный запрос.Я следовал этому превосходному учебнику:

http://dblog.com.au/iphone-development-tutorials/iphone-sdk-tutorial-reading-data-from-a-sqlite-database/

...