2011-10-08 23 views
5

Próbuję uzyskać wyraźny wynik z NSPredicate.NSFetchRequest z różnymi właściwościami

Mój kod:

NSFetchRequest *request = [[NSFetchRequest alloc] init]; 
    NSEntityDescription *entity = [NSEntityDescription entityForName:@"Members" inManagedObjectContext:context]; 
    request.entity = entity; 
    request.sortDescriptors = [NSArray arrayWithObject:[NSSortDescriptor sortDescriptorWithKey:@"city" 
                        ascending:YES 
                         selector:@selector(caseInsensitiveCompare:)]]; 

    request.predicate = [NSPredicate predicateWithFormat:@"memberDeleted == %@", [NSNumber numberWithBool:NO]]; 

    NSDictionary *properties = [entity propertiesByName]; 
    request.propertiesToFetch = [NSArray arrayWithObject:[properties objectForKey:@"city"]]; 
    request.returnsDistinctResults = YES; 

    request.fetchBatchSize = 20; 

    NSFetchedResultsController *frc = [[NSFetchedResultsController alloc] initWithFetchRequest:request 
                      managedObjectContext:context 
                      sectionNameKeyPath:nil 
                        cacheName:@"CityCache"]; 
    [request release]; 

    self.fetchedResultsController = frc; 
    [frc release]; 

Problem jest, że wynik powraca wiele razy w tym samym mieście. Ten podmiot ma wielu członków, a każdy poseł ma właściwość "miasto".

Co robię źle?

Dzięki

RL

Odpowiedz

7

Upewnij się, aby ustawić resultType z NSFetchRequest do NSDictionaryResultType. Wartością domyślną jest zwracanie rzeczywistych obiektów, przez co zostanie zignorowany propertiesToFetch.

+0

powinienem chyba dodawać, że 'NSFetchedResultsController' generalnie nie jest fanem' NSDictionaryResultType'. Możesz napotkać inne problemy i zrezygnować z 'NSFetchedResultsController'. – Alex

+0

Pytanie brzmi, że nie chcę, aby wynik był i NSDictionary. Jeśli nie, będę musiał zmienić część mojego kodu. Czy można to zrobić inaczej? –

+0

Jakiego obiektu oczekujesz, że powróci? Dane podstawowe mogą tylko zwracać obiekty, które faktycznie istnieją na wykresie obiektu. – Alex

6

Z pomocą @Alex, oto ostateczny kod, bez FRC:

NSFetchRequest *request = [[NSFetchRequest alloc] init]; 
NSEntityDescription *entity = [NSEntityDescription entityForName:@"Members" inManagedObjectContext:self.managedObjectContext]; 
request.entity = entity; 
request.sortDescriptors = [NSArray arrayWithObject:[NSSortDescriptor sortDescriptorWithKey:@"city" 
                       ascending:YES 
                        selector:@selector(caseInsensitiveCompare:)]]; 

request.predicate = [NSPredicate predicateWithFormat:@"memberDeleted == %@", [NSNumber numberWithBool:NO]]; 

NSDictionary *properties = [entity propertiesByName]; 
request.propertiesToFetch = [NSArray arrayWithObject:[properties objectForKey:@"city"]]; 
request.returnsDistinctResults = YES; 
request.resultType = NSDictionaryResultType; 
request.fetchBatchSize = 20; 

NSError *error = nil; 
NSArray *tempArray = [self.managedObjectContext executeFetchRequest:request error:&error]; 

NSMutableArray *cities = [[NSMutableArray alloc] init]; 
for (int i=0; i < [tempArray count]; i++){ 
    NSDictionary *tempDict = [NSDictionary dictionary]; 
    tempDict = [tempArray objectAtIndex:i]; 

    if ([tempDict objectForKey:@"city"] != nil) 
     [cities addObject:[tempDict objectForKey:@"city"]]; 
} 

//if the tempArray has no nil values, it's more efficient with: 
//NSArray* cities = [tempArray valueForKeyPath:@"city"]; 

self.cityArray = cities; 
[cities release]; 
[request release]; 

ta zwraca NSArray z listy miast.

Dzięki.

+1

Możesz to zrobić łatwiej. Po prostu wykonaj: 'NSArray * cities = [tempArray valueForKeyPath: @" city "];' Otrzyma to wartość 'city' dla każdego słownika w tablicy i umieści je w tablicy. – Alex

+1

@Alex, który działa poprawnie, jeśli w tablicy nie ma wartości zerowej. Jeśli jest jakaś wartość zerowa, ulega awarii. –

+0

Ups. Podczas rozpatrywania tego problemu wznowiłem komentarz Rui "crash", uważając go za cenną ostrożność. Ale właściwość ref klasy NSArray wymienia nieco inną metodę, "valueForKey:", która, jak mówi, zwróci obiekt NSNull dla obiektów zerowych. Więc nie powinno być problemu. – Wienke

2

Z oficjalnej dokumentacji:

NSManagedObjectContext *context = <#Get the context#>; 

NSEntityDescription *entity = [NSEntityDescription entityForName:@"<#Entity name#>" inManagedObjectContext:context]; 

NSFetchRequest *request = [[NSFetchRequest alloc] init]; 
[request setEntity:entity]; 
[request setResultType:NSDictionaryResultType]; 
[request setReturnsDistinctResults:YES]; 
[request setPropertiesToFetch :[NSArray arrayWithObject:@"<#Attribute name#>"]]; 

// Execute the fetch. 
NSError *error; 
id requestedValue = nil; 
NSArray *objects = [managedObjectContext executeFetchRequest:request error:&error]; 
if (objects == nil) { 
    // Handle the error. 
} 
Powiązane problemy