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];
}