在UITableViewController中使用UISearchDisplayController时断言失败

时间:2013-01-08 01:56:57

标签: ios ios6 uitableview uisearchdisplaycontroller

我一直在尝试将简单的搜索功能添加到我的应用中的TableViewController。我按照Ray Wenderlich的教程。我有一个带有一些数据的tableView,我在storyboard中添加了搜索栏+显示控制器,然后我有了这段代码:

#pragma mark - Table View
     - (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath{
        UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:@"BreedCell" forIndexPath:indexPath];

        //Create PetBreed Object and return corresponding breed from corresponding array
        PetBreed *petBreed = nil;

        if(tableView == self.searchDisplayController.searchResultsTableView)
            petBreed = [_filteredBreedsArray objectAtIndex:indexPath.row];
        else
            petBreed = [_breedsArray objectAtIndex:indexPath.row];

        cell.accessoryType  = UITableViewCellAccessoryDisclosureIndicator;
        cell.textLabel.text = petBreed.name;

        return cell;
    }

#pragma mark - Search
    -(BOOL)searchDisplayController:(UISearchDisplayController *)controller shouldReloadTableForSearchString:(NSString *)searchString {
        [_filteredBreedsArray removeAllObjects];
        NSPredicate *predicate = [NSPredicate predicateWithFormat:@"SELF.name contains[c] %@",searchString];
        _filteredBreedsArray = [[_breedsArray filteredArrayUsingPredicate:predicate] mutableCopy];

        return YES;
    }

    -(BOOL)searchDisplayController:(UISearchDisplayController *)controller shouldReloadTableForSearchScope:(NSInteger)searchOption {
        // Tells the table data source to reload when scope bar selection changes

        [_filteredBreedsArray removeAllObjects];
        NSPredicate *predicate = [NSPredicate predicateWithFormat:@"SELF.name contains[c] %@",self.searchDisplayController.searchBar.text];
        _filteredBreedsArray = [[_breedsArray filteredArrayUsingPredicate:predicate] mutableCopy];
        return YES;
    }

标准内容,但是当我在搜索栏中输入文本时,每次都会出现此错误:

2013-01-07 19:47:07.330 FindFeedo[3206:c07] *** Assertion failure in -[UISearchResultsTableView dequeueReusableCellWithIdentifier:forIndexPath:], /SourceCache/UIKit_Sim/UIKit-2372/UITableView.m:4460
2013-01-07 19:47:07.330 FindFeedo[3206:c07] *** Terminating app due to uncaught exception 'NSInternalInconsistencyException', reason: 'unable to dequeue a cell with identifier BreedCell - must register a nib or a class for the identifier or connect a prototype cell in a storyboard'

据我所知,在iOS 6中,单元格的处理和出列系统发生了变化,而且搜索使用了不同的tableView,所以我认为问题是带有过滤结果的搜索tableView不知道单元格,所以我把它放在我的viewDidLoad中:

[self.searchDisplayController.searchResultsTableView registerClass:[UITableViewCell class] forCellReuseIdentifier:@"BreedCell"];

瞧!它工作...只有你第一次搜索。如果您返回原始结果并再次搜索,则应用程序崩溃时会出现相同的错误。我想也许可以添加所有

if(!cell){//init cell here};

填充到cellForRow方法,但是这不符合dequeueReusableCellWithIdentifier:forIndexPath:方法的全部目的吗?无论如何,我迷路了。我错过了什么?请帮忙。提前感谢您的所有时间(:

亚历。

7 个答案:

答案 0 :(得分:193)

尝试在dequeueReusableCellWithIdentifier中使用self.tableView而不是tableView:

- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath{
    UITableViewCell *cell = [self.tableView dequeueReusableCellWithIdentifier:@"BreedCell"];

    //Create PetBreed Object and return corresponding breed from corresponding array
    PetBreed *petBreed = nil;

    if(tableView == self.searchDisplayController.searchResultsTableView)
        petBreed = [_filteredBreedsArray objectAtIndex:indexPath.row];
    else
        petBreed = [_breedsArray objectAtIndex:indexPath.row];

    cell.accessoryType  = UITableViewCellAccessoryDisclosureIndicator;
    cell.textLabel.text = petBreed.name;

    return cell;
}

此代码效果很好

注意

如果您有自定义高度单元格,请不要使用

[self.tableView dequeueReusableCellWithIdentifier:CellIdentifier forIndexPath:indexPath];

改为使用

[self.tableView dequeueReusableCellWithIdentifier:CellIdentifier];

答案 1 :(得分:14)

它首次运行时工作得很好但是如果你退出结果表然后又进行另一次搜索就会崩溃的原因是因为每次进入搜索模式时搜索显示控制器都会加载一个新的UITableView

通过搜索模式我的意思是,您已经点击了文本字段并且您已经开始输入,此时会生成一个表格视图以显示结果,通过点击取消按钮退出此模式。当您第二次点击文本字段并再次开始输入时 - 这是第二次进入“搜索模式”。

因此,为了避免崩溃,您应该为表视图注册单元类,以便在searchDisplayController:didLoadSearchResultsTableView:委托方法(来自UISearchDisplayDelegate)而不是控制器viewDidLoad中使用方法

如下:

- (void)searchDisplayController:(UISearchDisplayController *)controller didLoadSearchResultsTableView:(UITableView *)tableView
{
    [tableView registerClass:[DPContentTableCell class] forCellReuseIdentifier:cellIdentifier];
    [tableView registerClass:[DPEmptyContentTableCell class] forCellReuseIdentifier:emptyCellIdentifier];
}

这让我感到惊讶,因为在iOS 7上......表视图正在被重用。因此,如果您愿意,可以在viewDidLoad注册课程。对于遗产,我会在我提到的委托方法中保留我的注册。

答案 2 :(得分:11)

搜索后,' tableView' cellForRowAtIndexPath方法似乎不是您定义的表的实例。因此,您可以使用定义单元格的表的实例。而不是:

UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:@"Cell" forIndexPath:indexPath];

使用:

UITableViewCell *cell = [self.tableView dequeueReusableCellWithIdentifier:@"Cell" forIndexPath:indexPath];

(不要使用 tableView 的cellForRowAtIndexPath方法,使用 self.tableView 。)

答案 3 :(得分:3)

不使用'indexPath'使单元格出列,如果获得nil元素,则必须手动分配。

- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
{
    UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:@"YourCellId"];
    if (!cell)
        cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:@"YourCellId"];

    // fill your cell object with useful stuff :)

    return cell;
}

当您有一个分段主列表和一个普通搜索列表时,尝试使用self.tableView将该单元格出列可能会导致崩溃。 此代码可以在任何情况下使用。

答案 4 :(得分:3)

当我遇到这个问题时,解决方案是用tableView替换self.tableView dequeueReusableCellWithIdentifier:@yourcell

答案 5 :(得分:2)

我正在制作该教程。默认的TableViewController具有“forIndexPath”,在他的示例中它不存在。删除后,搜索就可以了。

//Default code
UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:CellIdentifier forIndexPath:indexPath];

//Replace with
UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:CellIdentifier];

答案 6 :(得分:2)

对于swift 3,您只需要添加self:

func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell
{
    let cell = self.tableView.dequeueReusableCell(withIdentifier: "yourCell", for: indexPath) as! YourCell

    ...
}