慎用 reloadRowsAtIndexPaths

banner.jpg

reloadRowsAtIndexPaths 一般用于刷新一组 cell,笔者在使用过程中发现,调用该方法后 Tableview 并不是刷新 cell,而是会重新创建 cell。

并且 cell 的重用会发生错乱,在 IndexPath [0, 0] 下创建的 cell 会在 IndexPath [0, 1] 下被重用。

比如一个常见的需求,在 TableView 第一行显示头像,第二行显示昵称。假设这两条信息是单独从服务器获取的。

normal.png

模拟网络请求代码如下:


- (void)viewDidLoad {
    [super viewDidLoad];
    _titles = @[@"Avator", @"Nickname"];
    [self.tableView registerClass:[SFTableViewCell class] forCellReuseIdentifier:kCellIdentifier];
    
    dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(kDelayTime * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
    
        _avator = [UIImage imageNamed:@"cat"];
        NSIndexPath *indexPath = [NSIndexPath indexPathForRow:0 inSection:0];
        [self.tableView reloadRowsAtIndexPaths:@[indexPath] withRowAnimation:UITableViewRowAnimationNone];
        
        dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(kDelayTime * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
        
            _nickName = @"Smallfly";
            NSIndexPath *indexPath = [NSIndexPath indexPathForRow:1 inSection:0];
            [self.tableView reloadRowsAtIndexPaths:@[indexPath] withRowAnimation:UITableViewRowAnimationNone];
            
        });
    });
}

但实际的的显示结果却是这样:

actual.png

再来看一下 TableView DataSource 的实现:

- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
    // Why reloadRowsAtIndexPaths [0,0] returned cell is nil?
    SFTableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:kCellIdentifier forIndexPath:indexPath];
    cell.textLabel.text = _titles[indexPath.row];

    if (indexPath.row == 0) {
        [self configureCell:cell indexPath:indexPath];
    } else {
        NSString *nickName =  _nickName ?: @"nickname";
        cell.detailTextLabel.text = nickName;
    }
    return cell;
}

在 indexPath.row == 0 的时候为 cell 添加头像:

- (void)configureCell:(UITableViewCell *)cell indexPath:(NSIndexPath *)indexPath {
    
    for (UIView *sv in cell.contentView.subviews) {
        if ([sv isKindOfClass:[UIImageView class]]) {
            [sv removeFromSuperview];
        }
    }
    
    UIImage *avator = _avator ?: [UIImage imageNamed:@"user_profile_avatar"];
    UIImageView *avatorImageView = [[UIImageView alloc] initWithImage:avator];
    // configure imageView...
    [cell.contentView addSubview:avatorImageView];
}

为防止重复的添加 imageView ,先将存在的头像移除。

进行一番调试之后笔者发现,在请求前,两个 cell 的地址为:

<SFTableViewCell: 0x7fb1bf826c00> // indexPath [0, 0]
<SFTableViewCell: 0x7fb1be00fc00> // indexPath [0, 1]

在请求后调用 reloadRowsAtIndexPaths 两个 cell 的地址为:

<SFTableViewCell: 0x7fb1be001600> // indexPath [0, 0]
<SFTableViewCell: 0x7fb1bf826c00> // indexPath [0, 1]

存在诡异的地方:

  1. 在 reloadRowsAtIndexPaths 是 indexPath [0, 0] 的 cell 被重新创建了。
  2. indexPath [0, 1] cell 地址等于 indexPath [0, 0] cell 的地址。

因为以上两个问题,导致在刷新第二行 cell 时出现了原本在第一行的默认头像,如果用 reload 刷新整个 TableView 则不会有这样的问题。

当然,最好的方式是自定义头像 cell,为方便起见使用这种方式,发现了这个问题。如果有大神知道原理,请指点一二 。

参考 Demo 地址 UITableViewBug

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,384评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,845评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,148评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,640评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,731评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,712评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,703评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,473评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,915评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,227评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,384评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,063评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,706评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,302评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,531评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,321评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,248评论 2 352

推荐阅读更多精彩内容