UITableView重用机制主要是依靠reuseIdentifier来辨别,以此来建立一个队列,将建好的Cell放入队列中,之后直接使用队列中的Cell,不再新建,极大的提升了TableView的重用性,同时使列表滑动时不会出现卡顿现象。TableView基本上是新手必学,我第一个熟练掌握的控件,及使用得最多的就是TableVIew,重用机制有很多地方很多人都讲过,不过我还是要从自己的角度来再次BB下。由于列表内容太长了,不方便直接在界面上截图,所以只把打印的数据拿出来。

1、重用机制Cell是如何新建的?

很多人以为Cell在新建的时候只新建一个,其他的都重复使用那一个,但事实上并不是这样的,新建多少个Cell,并加入重用队列是取决于初始化时屏幕中能容纳多少个Cell,(5S模拟器测试)经测试Cell高44时,新建14个Cell。Cell高120时,count:6,新建6个Cell。


- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
    static int count = 0;
    static NSString *celleIdentifier = @"Cell";

    UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:celleIdentifier];
    if (cell == nil) {
        cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:celleIdentifier];
        cell.selectionStyle = UITableViewCellSelectionStyleGray;
        count ++;
    }
    if (indexPath.row < 10) {
        cell.textLabel.text = [NSString stringWithFormat:@"%ld",(long)indexPath.row];
    
    NSLog(@"count:%d,cellheight:%f",count,cell.frame.size.height);
    return cell;
}



iOS中cell复用赋值错乱该怎么解决 tableview复用机制_uitableview

iOS中cell复用赋值错乱该怎么解决 tableview复用机制_数据_02


2、Cell重用时是如何取出来的?

在使用120高度的Cell时,重用队列中将只有6个存储的Cell,在下方代码中到第6个Cell,即textLabel == 6时 已用开始使用重用队列中的Cell,由打印的数据可知,init textLabel为0,这使用的是第一个存入的队列Cell,textLabel == 7时, init textLabel为1,使用的是存入的第二个Cell,一直到使用第六个完成一轮后再重新使用第一个存入的Cell,由此可以推断,Cell重用时是秉承先进先出的原则,循环使用。


- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
    static int count = 0;
    static NSString *celleIdentifier = @"Cell";
    UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:celleIdentifier];
    if (cell == nil) {
        cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:celleIdentifier];
        cell.selectionStyle = UITableViewCellSelectionStyleGray;
        count ++;
    }
    NSLog(@"init textLabel:%@",cell.textLabel.text);
    if (indexPath.row < 10) {
        cell.textLabel.text = [NSString stringWithFormat:@"%ld",(long)indexPath.row];
    }
    NSLog(@"count:%d,textLabel:%@",count,cell.textLabel.text);
    return cell;
}



iOS中cell复用赋值错乱该怎么解决 tableview复用机制_iOS中cell复用赋值错乱该怎么解决_03


3、reuseIdentifier的作用?

在加载第11个Cell的时候开始,Cell的celleIdentifier换成了@"cell2",并且将第11个Cell的textLabel内容换乘@"this is 20",根据截图可知,由于更换了celleIdentifier,在第11个Cell加载时是没有init textLabel的,并且count开始自增直至12,说明使用新的celleIdentifier使tableview重新建了一个队列并且放入了6个新的Cell,并且在6个一轮之后重用了,新队列里的Cell。

- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
    static int count = 0;
    static NSString *celleIdentifier = @"Cell";
    if (indexPath.row > 10) {
        celleIdentifier = @"Cell2";
    }
    UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:celleIdentifier];
    if (cell == nil) {
        cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:celleIdentifier];
        cell.selectionStyle = UITableViewCellSelectionStyleGray;
        count ++;
    }
    NSLog(@"init textLabel:%@",cell.textLabel.text);
    if (indexPath.row < 10) {
        cell.textLabel.text = [NSString stringWithFormat:@"%ld",(long)indexPath.row];
    }else if (indexPath.row == 11){
        cell.textLabel.text = @"this is 20";
    }
    NSLog(@"count:%d,textLabel:%@",count,cell.textLabel.text);
    return cell;
}



iOS中cell复用赋值错乱该怎么解决 tableview复用机制_iOS中cell复用赋值错乱该怎么解决_04



4、为什么使用重用机制后我的数据乱掉了?

在解释第三点时,在第11个Cell之后textLabel一直显示的是null,这是因为我没有在重用后,对这个Cell进行操作,一般在使用重用后,都需要对Cell的数据进行处理。如果没有处理的话,一种情况就像上面截图一样,由于重用队列里面的Cell的textLabel就是没有值的,取出来之后还是不进行操作,即为null。还有一种情况是打印出this is 20的这种情况,因为重用队列里面的Celld的textLabel里面有值,取出重用后不进行操作的话就会使用之前队列里Cell的textLabel这个值,所以后面有好几个重用Cell2队列中第一个存入的Cell时,打印出来的数据也为this is 20。


总结:不同的reuseIdentifier代表了不同的队列,重用时按队列顺序先进先出,一轮之后再重新从第一个开始重用。