Tableview使用空数组获取索引超出范围

Posted

技术标签:

【中文标题】Tableview使用空数组获取索引超出范围【英文标题】:Tableview get index out of bounds with empty array 【发布时间】:2013-04-10 05:09:42 【问题描述】:

我有一个tableview,里面有一个fetchedresultscontroller 和一个searchbar。到目前为止,在创建应用程序时,我只使用了 7 个单元格并且工作得很好,但是自从我开始使用超过 7 个单元格后,当我触摸搜索栏时,我得到了超出范围空数组的错误。但是当我把它放回 7 个单元格时,它工作得很好。

可能出了什么问题?

错误: 由于未捕获的异常“NSRangeException”而终止应用程序,原因:“*-[__NSArrayM objectAtIndex:]: 索引 2 超出空数组的范围”

- (void)viewDidLoad

    [super viewDidLoad];

    self.title = @"Exercises";



    id appDelegate = [[UIApplication sharedApplication] delegate];
    _managedObjectContext = [appDelegate managedObjectContext];

    NSError *error;
    if (![[self fetchedResultsController] performFetch:&error]) 
        NSLog(@"%@ %@", error, [error userInfo]);
        abort();
    

    _searchResults = [[NSMutableArray alloc] init];




- (void)viewDidAppear:(BOOL)animated

    [super viewDidAppear:animated];

    NSIndexPath *selectedIndexPath = [_tableView indexPathForSelectedRow];
    if (!selectedIndexPath) 
        [_tableView setContentOffset:CGPointMake(0, 44) animated:NO];
    
    else 
        [_tableView deselectRowAtIndexPath:selectedIndexPath animated:YES];
    


- (void)viewDidDisappear:(BOOL)animated

    [super viewDidDisappear:animated];
    if (self.searchDisplayController.isActive) 
        [self.searchDisplayController setActive:NO animated:YES];
    


- (void)didReceiveMemoryWarning

    [super didReceiveMemoryWarning];


#pragma mark - UITableViewDataSource

- (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView

    if (self.searchDisplayController.isActive) 
        return 1;
    
    return [[_fetchedResultsController sections] count];


- (NSString *)tableView:(UITableView *)tableView titleForHeaderInSection:(NSInteger)section

    if (self.searchDisplayController.isActive) 
        return nil;
    
    return [[[_fetchedResultsController sections] objectAtIndex:section] name];


- (UIView *)tableView:(UITableView *)tableView viewForHeaderInSection:(NSInteger)section

    UIView *view = [[UIView alloc] initWithFrame:CGRectMake(0, 0, self.view.bounds.size.width, 22)];
    view.backgroundColor = [UIColor colorWithRed:20.0/255.0 green:20.0/255.0 blue:20.0/255.0 alpha:0.8];

    UILabel *sectionLabel = [[UILabel alloc] initWithFrame:CGRectMake(20, 0, 160, 22)];
    sectionLabel.font = [UIFont fontWithName:@"Avenir-Book" size:16.0f];
    sectionLabel.text = @"Search Results";

    if (!self.searchDisplayController.isActive) 
        sectionLabel.text = [[[_fetchedResultsController sections] objectAtIndex:section] name];
    

    sectionLabel.backgroundColor = [UIColor clearColor];
    sectionLabel.textColor = [UIColor whiteColor];
    [view addSubview:sectionLabel];
    return view;


- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section

    if (self.searchDisplayController.isActive) 
        return [_searchResults count];
    
    return [[[_fetchedResultsController sections] objectAtIndex:section] numberOfObjects];


- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath

    static NSString *cellIdentifier = @"exerciseCell";
    UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:cellIdentifier];

    if (!cell) 
        cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:cellIdentifier];
        [self stampCell:cell atIndexPath:indexPath];
    

    [self configureCell:cell atIndexPath:indexPath];

    return cell;



#pragma mark - Cell

- (void)stampCell:(UITableViewCell *)cell atIndexPath:(NSIndexPath *)indexPath

    if ([cell.reuseIdentifier isEqualToString:@"exerciseCell"]) 

        UILabel *textLabel = [[UILabel alloc] initWithFrame:CGRectMake(70, 2, 150, 22)];
        textLabel.tag = kExerciseCellTextLabel;
        textLabel.font = [UIFont systemFontOfSize:18.0];
        textLabel.textColor = [UIColor blackColor];
        textLabel.highlightedTextColor = [UIColor whiteColor];
        textLabel.backgroundColor = [UIColor clearColor];
        [cell.contentView addSubview:textLabel];

        UILabel *detailTextLabel = [[UILabel alloc] initWithFrame:CGRectMake(70, 24, 150, 18)];
        detailTextLabel.tag = kExerciseCellDetailTextLabel;
        detailTextLabel.font = [UIFont systemFontOfSize:14.0];
        detailTextLabel.textColor = [UIColor colorWithRed:128.0/255.0 green:128.0/255.0 blue:128.0/255.0 alpha:1.0];
        detailTextLabel.highlightedTextColor = [UIColor colorWithRed:127.0/255.0 green:127.0/255.0 blue:127.0/255.0 alpha:1.0];
        detailTextLabel.backgroundColor = [UIColor clearColor];
        [cell.contentView addSubview:detailTextLabel];

    


- (void)configureCell:(UITableViewCell *)cell atIndexPath:(NSIndexPath *)indexPath

    if ([cell.reuseIdentifier isEqualToString:@"exerciseCell"]) 

        UILabel *textLabel = (UILabel *)[cell.contentView viewWithTag:kExerciseCellTextLabel];
        UILabel *detailTextLabel = (UILabel *)[cell.contentView viewWithTag:kExerciseCellDetailTextLabel];

        Exercise *exercise;

        if (self.searchDisplayController.isActive) 
            exercise = [_searchResults objectAtIndex:indexPath.row];
        
        else 
            exercise = [_fetchedResultsController objectAtIndexPath:indexPath];
        

        textLabel.text = [exercise valueForKey:kExerciseName];
        detailTextLabel.text = [NSString stringWithFormat:@"%@ - %@", [[exercise valueForKey:kExerciseEType] valueForKey:kExerciseTypeName], [[exercise valueForKey:kExerciseGear] valueForKey:kGearName]];

        cell.selectionStyle = UITableViewCellSelectionStyleGray;
        cell.accessoryType = UITableViewCellAccessoryDisclosureIndicator;

    



#pragma mark - UITableViewDelegate

- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath

    [self performSegueWithIdentifier:@"exerciseDetail" sender:self];


#pragma mark - UISearchBarDelegate

- (BOOL)searchDisplayController:(UISearchDisplayController *)controller shouldReloadTableForSearchString:(NSString *)searchString

    [_searchResults removeAllObjects];
    if (searchString.length > 0) 

        for (Exercise *exercise in [_fetchedResultsController fetchedObjects]) 
            NSRange range = [[exercise valueForKey:kExerciseName] rangeOfString:searchString options:NSCaseInsensitiveSearch];
            if (range.location == 0) 
                [_searchResults addObject:exercise];
            
        
    
    return YES;


- (BOOL)searchBarShouldBeginEditing:(UISearchBar*)searchBar 
    [super setPan];
    return YES;


- (BOOL)searchBarShouldEndEditing:(UISearchBar *)searchBar

    [super setPan];
    return YES;


#pragma mark - NSFetchedResultsControllerDelegate

- (NSFetchedResultsController *)fetchedResultsController

    if (_fetchedResultsController) 
        return _fetchedResultsController;
    

    NSFetchRequest *request = [[NSFetchRequest alloc] init];
    NSEntityDescription *entity = [NSEntityDescription entityForName:kExercise inManagedObjectContext:_managedObjectContext];
    [request setEntity:entity];

    NSSortDescriptor *sort = [NSSortDescriptor sortDescriptorWithKey:kExerciseName ascending:YES];
    [request setSortDescriptors:[NSArray arrayWithObject:sort]];

    NSFetchedResultsController *fetchedResultsController = [[NSFetchedResultsController alloc] initWithFetchRequest:request managedObjectContext:_managedObjectContext sectionNameKeyPath:kExerciseFirstLetter cacheName:@"ExerciseList"];
    [self setFetchedResultsController:fetchedResultsController];
    [_fetchedResultsController setDelegate:self];

    return _fetchedResultsController;


- (void)controllerWillChangeContent:(NSFetchedResultsController *)controller

    [self.tableView beginUpdates];


- (void)controller:(NSFetchedResultsController *)controller didChangeSection:(id <NSFetchedResultsSectionInfo>)sectionInfo
           atIndex:(NSUInteger)sectionIndex forChangeType:(NSFetchedResultsChangeType)type

    switch(type) 
        case NSFetchedResultsChangeInsert:
            [self.tableView insertSections:[NSIndexSet indexSetWithIndex:sectionIndex] withRowAnimation:UITableViewRowAnimationFade];
            break;

        case NSFetchedResultsChangeDelete:
            [self.tableView deleteSections:[NSIndexSet indexSetWithIndex:sectionIndex] withRowAnimation:UITableViewRowAnimationFade];
            break;
    


- (void)controller:(NSFetchedResultsController *)controller didChangeObject:(id)anObject
       atIndexPath:(NSIndexPath *)indexPath forChangeType:(NSFetchedResultsChangeType)type
      newIndexPath:(NSIndexPath *)newIndexPath

    UITableView *tableView = self.tableView;

    switch(type) 
        case NSFetchedResultsChangeInsert:
            [tableView insertRowsAtIndexPaths:@[newIndexPath] withRowAnimation:UITableViewRowAnimationFade];
            break;

        case NSFetchedResultsChangeDelete:
            [tableView deleteRowsAtIndexPaths:@[indexPath] withRowAnimation:UITableViewRowAnimationFade];
            break;

        case NSFetchedResultsChangeUpdate:
            [self configureCell:[tableView cellForRowAtIndexPath:indexPath] atIndexPath:indexPath];
            break;

        case NSFetchedResultsChangeMove:
            [tableView deleteRowsAtIndexPaths:@[indexPath] withRowAnimation:UITableViewRowAnimationFade];
            [tableView insertRowsAtIndexPaths:@[newIndexPath] withRowAnimation:UITableViewRowAnimationFade];
            break;
    


- (void)controllerDidChangeContent:(NSFetchedResultsController *)controller

    [self.tableView endUpdates];


@end

【问题讨论】:

我编辑了它,并添加了触摸搜索栏时触发的错误。 【参考方案1】:

它显示一些数组仍然是空的..检查它的更好方法

-将 NSLog 语句放在数组元素之后并运行它。它会显示哪个数组仍然为空。

这样做如果您的问题没有得到解决..请告诉我..这样,我可以进一步帮助您..

【讨论】:

你检查这个链接..它可能对你有帮助..***.com/questions/4965827/… 哪里是放置 nslog 的有效位置?或者位置无关紧要。 在视图 didload 中对所有获取的对象进行了快速枚举,以检查数组中是否有任何对象为 nil,但没有一个为 nil。你是这个意思吗? @JakeOrtiz k..让我再检查一次..您还将 NSLog 放在获取行详细信息的位置.. 放在configurecell方法中可以吗?【参考方案2】:

您在哪里看到崩溃 - 仅显示搜索结果或显示整个列表时?

如果只是搜索结果,那么您的 _searchResults 记录不超过 7 条。

如果是显示所有结果时,请检查 _fetchedResultsController.sections 的每个值中的记录数。看起来他们的记录不超过 7 条。

【讨论】:

如果我有 8 个或更多单元格并触摸它崩溃的搜索栏,它甚至不会显示 searchdisplaycontroller。而且它不需要对searchResults做任何事情,当搜索栏被触摸时它甚至不调用uitableviewdatasource。【参考方案3】:

您好,我想您从 _searchResults 中删除所有元素可能是因为此应用程序崩溃了。请检查一次。

- (BOOL)searchDisplayController:(UISearchDisplayController *)controller shouldReloadTableForSearchString:(NSString *)searchString

    [_searchResults removeAllObjects];

【讨论】:

我应该删除那行吗?

以上是关于Tableview使用空数组获取索引超出范围的主要内容,如果未能解决你的问题,请参考以下文章

SWIFT:为 tableView 计算 numberOfRowsInSection 时,数组索引超出范围

致命错误:TableView 中的数组索引超出范围

索引 0 超出空数组的界限,有时有效,有时无效

Swift 5:从 tableView numberOfRowsInSection 返回数组计数时索引超出范围

致命错误:索引超出范围的 tableview 有两个部分

致命错误:索引超出范围tableview