使用NSPredicate在两个数组中匹配id

时间:2010-06-23 07:30:51

标签: iphone objective-c nspredicate

您好我有一个简单的问题,但我无法处理它。

  • 我有一个Modelclass'Location',它包含一个带有类别ID的数组(12,23,56)。
  • 然后我有一个包含所有可用类别ID(1,2,3,4,5,6,7,...)
  • 的数组
  • 所有这些类别都有一个ID显示在TableView中,并且能够选择与否。
  • 我在MapView(注释类是位置)上显示了一个标记,它应该根据所提到的Filter TableView中过滤器的选择显示。

我需要实现一个“按类别筛选”功能,该功能会删除所有标记并再次添加它们,但这只是基于列表中的选择。

所以我需要将数组与位置模型中的filter-id进行比较,并将数组与TableView中的所有Filter-ID进行比较。我使用了以下功能:

for (Location *currLocation in arrListOfLocationsNearby) {

for(NSrtring * currKatId in currLocation.arrKatIds){

NSInteger intCurrKatId = [currKatId integerValue];
NSLog(@"Current KatId %@ id: %d \n", currLocation.strAdr, intCurrKatId);

for (Filter *currFilter in [SDM getSharedDataManager].arrListOfFilterOptions) {

 if (currFilter.isSelected) {

  NSInteger intCurrFilterId = [currFilter.strAtt_id intValue];
  NSLog(@"Current Filter %@ id: %d \n", currFilter.strAtt_text, intCurrFilterId);

  if ([currKatId intValue] == [currFilter.strAtt_id intValue]) {
   currLocation.isVisible = YES;
  }else {
   currLocation.isVisible = NO;
  }
 }
}

}   }

我知道这将是循环通过一切的最无效方式。我想以某种方式使用NSPredicate,但我以前从未使用它们,我找不到我的问题的例子。

你们的任何提示?

关于m。

3 个答案:

答案 0 :(得分:6)

所以每个“Location”对象都有很多“categoryIDs”?并且您想要在“arrayOfEnabledCategoryIDs”中显示任何具有categoryID的Location对象?

假设您有一个Location实体和一个CategoryID实体,并且它们之间存在多对多关系(locations <<--->> categories),您可以这样做:

NSArray * arrayOfEnabledCategoryIDs = ...; //these are the categories to display
NSFetchRequest * f = [[NSFetchRequest alloc] init];
[f setEntity:theEntityDescriptionForLocationObjects];
[f setPredicate:[NSPredicate predicateWithFormat:@"ANY categories.categoryID IN %@", arrayOfEnabledCategoryIDs]];

答案 1 :(得分:0)

尝试使用每个测试的字符串表示形式:

NSArray *arrayOfPredicateStrings = ;// array of all tests, probably [SDM sharedDataManager].arrListOfFilterOptions

NSPredicate *enabled = [NSPredicate predicateWithFormat:@"isSelected == true"];

NSArray *arrayOfEnabledPredicateStrings = 
        [arrayOfPredicateStrings filteredArrayUsingPredicate:enabled];

NSMutableString *formatString;
BOOL firstTime = YES;
for (NSString *string in arrayOfEnabledPredicateStrings) { 
    // Concatinate strings with OR inbetween
    if (firstTime) {
        firstTime = NO;
        formatString = [string mutableCopy];
    } else {
        [formatString appendFormat:@" OR %@", string]; 
    }
} // formatString should look something like @"ivar1 >= 1 OR ivar2 == true OR ivar3 != \"Hello, World\""
NSPredicate *predicate = [NSPredicate predicateWithFormat:string];

NSArray *arrListOfLocationsNearby; // assume already filled

NSArray *suitableNearbyLocations = 
        [arrListOfLocationsNearby filteredArrayUsingPrediacate:predicate];

如果CoreData数据存储位于SDM类后面,只需创建一个如下所示的获取请求:

NSPredicate *predicate = ;// get the predicate as above
NSManagedObjectContext  *context = ;// a managed object context
NSEntityDescription *entity = 
        [NSEntityDescription entityForName:@"entityName" inManagedObjectContext:context];

NSFetchRequest *request = [[NSFetchRequest alloc] init];
request.entity = entity;
request.predicate = predicate;

NSError *error;
NSArray *results = [context executeFetchRequest:request error:&error];
[request release];

可以找到谓词格式语法here,并且可以找到here

的用法

答案 2 :(得分:0)

如果您熟悉Ruby On Rails中的ActiveRecord。我一直在使用一个名为activerecord + fetching + for + core的优秀库,你可以在这里找到更多关于它的信息http://github.com/magicalpanda/activerecord-fetching-for-core-data

NSArray *departments = [NSArray arrayWithObjects:dept1, dept2, ..., nil];
NSPredicate *peopleFilter = [NSPredicate predicateWithFormat:@"Department IN %@", departments];

NSArray *people = [Person findAllWithPredicate:peopleFilter];

它确实清理了我的代码。