NSPredicate: получить один из каждого вида

Я хочу создать NSFetchRequest для таких объектов:

The Object и Car что есть attribute color. У меня четыре машины:--9-->

car1.color = red
car2.color = red
car3.color = blue
car4.color = green

Я хочу создать NSPredicate это выбирает только один автомобиль для каждого цвета(не имеет значения, какой автомобиль он выбирает.

как я могу этого достичь?

на самом деле я ищу что-то похожее на DISTINCT в SQL

5 ответов


Основные Данные тут поддержка выборки различных значений. Apple даже предоставляет пример кода здесь: http://developer.apple.com/library/ios/#documentation/DataManagement/Conceptual/CoreDataSnippets/Articles/fetchExpressions.html


страницы:

Получение Различных Значений

чтобы получить уникальные значения определенного атрибута по всем экземпляры данной сущности, вы настраиваете выборку запрос метод setReturnsDistinctResults: (и передайте YES в качестве параметра). Вы также укажите, что fetch должен возвращать словари, а не управляемые объекты и имя свойства, которое вы хотите получить.

Swift 2.0

let context:NSManagedObjectContext! = <#Get the context#>
let entity = NSEntityDescription.entityForName( "<#Entity name#>",  inManagedObjectContext:context )
let request = NSFetchRequest()
request.entity = entity
request.resultType = .DictionaryResultType
request.returnsDistinctResults = true
request.propertiesToFetch = [ "<#Attribute name#>" ]

var objects:[[String:AnyObject]]?
do
{
    try objects = context.executeFetchRequest( request )
}
catch
{
    // handle failed fetch
}

Obj-C

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:@[@"<#Attribute name#>"]];  

// Execute the fetch. 

NSError *error = nil; 
id requestedValue = nil; 
NSArray *objects = [context executeFetchRequest:request error:&error]; 
if (objects == nil) {
    // Handle the error. 
} 

(один "gotcha" заключается в том, что результаты будут возвращены как словари-если вам нужны объекты, вы можете сделать другую выборку, чтобы получить их ID)


вы пытались использовать valueForKeyPath ? Вы можете сделать что-то подобное в массиве, который предоставляет вам NSPredicate.

NSArray* distinctColors = [cars valueForKeyPath:@"@distinctUnionOfObjects.color"];

он вернет вам различные цвета, которые у вас есть, я не знаю, если это то, что вы хотите, но с цветами вы можете сделать что-то вроде @Anupdas сказал:


если вы ориентируетесь на iOS 4.0 или более поздней версии и не используете хранилище SQL Core-Data, Почему бы не использовать predicateWithBlock:?

следующее создаст NSFetchRequest вы хотите.

- (NSFetchRequest*) fetchRequestForSingleInstanceOfEntity:(NSString*)entityName groupedBy:(NSString*)attributeName
{
  __block NSMutableSet *uniqueAttributes = [NSMutableSet set];

  NSPredicate *filter = [NSPredicate predicateWithBlock:^(id evaluatedObject, NSDictionary *bindings) {
    if( [uniqueAttributes containsObject:[evaluatedObject valueForKey:attributeName]] )
      return NO;

    [uniqueAttributes addObject:[evaluatedObject valueForKey:attributeName]];
    return YES;
  }];

  NSFetchRequest *req = [NSFetchRequest fetchRequestWithEntityName:entityName];
  req.predicate = filter;

  return req;
}

затем вы можете создать новый метод для выполнения выборки и возврата результатов, которые вы хотите.

- (NSArray*) fetchOneInstanceOfEntity:(NSString*)entityName groupedBy:(NSString*)attributeName
{
  NSFetchRequest *req = [self fetchRequestForSingleInstanceOfEntity:entityName groupedBy:attributeName];

  // perform fetch
  NSError *fetchError = nil;
  NSArray *fetchResults = [_context executeFetchRequest:req error:&fetchError];

  // fetch results
  if( !fetchResults ) {
    // Handle error ...
    return nil;
  }

  return fetchResults;
}

попробуйте это:

NSFetchRequest *request = [[NSFetchRequest alloc] init];
NSEntityDescription *entity =
[NSEntityDescription entityForName:@"Selected_data"
            inManagedObjectContext:self.managedObjectContext];
[request setEntity:entity];
NSPredicate *predicatecnName = [NSPredicate predicateWithFormat:@"countries contains[cd] %@", Countryid];
NSPredicate *predicateLn = [NSPredicate predicateWithFormat:@"languages contains[cd] %@", languageid];
NSArray *subPredicates = [NSArray arrayWithObjects:predicatecnName, predicateLn, nil];

NSPredicate *predicate = [NSCompoundPredicate andPredicateWithSubpredicates:subPredicates];


[request setPredicate:predicate];
NSMutableArray *sendfd=[[NSMutableArray alloc] init];
NSError *error;
NSArray *array1 = [self.managedObjectContext executeFetchRequest:request error:&error];


if (array1 != nil) {


    for (Selected_data *sld in array1)
    {

        [sendfd addObject:sld.providers];
    }


}
else {

}
return sendfd;

попробуй это:

    NSFetchRequest *fetchRequest = [NSFetchRequest fetchRequestWithEntityName:@"Cars"];
    NSArray *arrayValues = [self.managedObjectContext executeFetchRequest:fetchRequest error:&error];

if (arrayValues.count != 0) {
     NSDictionary *result = [NSDictionary dictionaryWithObjects:arrayValues
                           forKeys:[arrayValues valueForKey:@"color"]];
     return [result allValues];
}