80

我一直在尝试向我的应用程序中的 TableViewController 添加简单的搜索功能。我遵循了 Ray Wenderlich 的教程。我有一个带有一些数据的 tableView,我在情节提要中添加了搜索栏 + 显示控制器,然后我有了以下代码:

#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: 方法的全部目的吗?无论如何,我迷路了。我错过了什么?请帮忙。提前感谢您的所有时间(:

亚历克斯。

4

7 回答 7

194

尝试在 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];
于 2013-01-08T18:50:38.960 回答
14

它在第一次运行时运行良好但如果您退出结果表并返回进行另一次搜索时崩溃的原因是因为UITableView每次进入搜索模式时搜索显示控制器都会加载一个新的。

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

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

如下:

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

这让我感到意外,因为在 iOS 7 上……表格视图正在被重用。因此,您可以根据需要注册课程viewDidLoad。为了遗产,我将在我提到的委托方法中保留我的注册。

于 2013-08-14T09:48:25.567 回答
12

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

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

采用:

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

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

于 2014-07-12T18:30:24.890 回答
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 使单元格出列可能会导致崩溃。相反,此代码适用于任何情况。

于 2013-04-24T14:51:06.403 回答
3

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

于 2014-10-02T20:23:50.283 回答
2

我也在研究那个教程。默认的 TableViewController 具有“forIndexPath”,在他的示例中它不存在。一旦我删除它,搜索就会起作用。

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

//Replace with
UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:CellIdentifier];
于 2013-09-10T15:49:35.153 回答
2

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

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

    ...
}
于 2017-04-18T17:43:35.810 回答