UITableView的cell的释放、内容消失等的思考总结

首先问题是在tableview的某个cell上面放了一个UITextField,然后textField填写的东西划出屏幕,再回来内容就消失了。后来知道是cell被替换(不是释放)了,然后又重新构建了一个textField,已经不是以前的那个,所以肯定是空的。不过这里倒是有个好玩的,就是如果你正在编辑某个textField,你把它划出屏幕,这个textfield所在的cell是不会释放的,内容也还在,不过这点倒是对整体没有帮助。关于UITableView会释放他的划出屏幕的cell,有几点总结:1、这个跟重用没关系,不知是从别人那里看来的还是我自己一开始就想错了,以为使用了UITableView的重用机制,它的cell才会在划出屏幕的时候被释放,其实不使用它也会释放。我的实验思路是在不使用重用的时候,添加一个__weak
修饰的UITableViewCell指针 成员变量,然后在tableViewCell构建的时候,让它指向某个cell,然后在cell构建之前输入它。

<span style="font-size:18px;">@interface ViewController (){
    __weak UITableViewCell* _testCell;

}</span>

<span style="font-size:18px;">-(UITableViewCell*)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath{
    UITableViewCell* cell = nil;

    if (indexPath.row == 1) {
        NSLog(@"_testCell >>> %@",_testCell);
        if (_testCell != nil) {
            return _testCell;
        }
    }

    if (cell == nil) {
        cell = [[UITableViewCell alloc]initWithStyle:(UITableViewCellStyleDefault) reuseIdentifier:@"Cell"];
        UITextField* textField = [[UITextField alloc]initWithFrame:CGRectMake(10, 0, 100, 44)];
        textField.tag = 1000;
        textField.borderStyle = UITextBorderStyleRoundedRect;
        textField.text = @"XXX";

        [cell addSubview:textField];
    }

    if (indexPath.row == 1) {<span style="white-space:pre">	</span>//引用第2个cell,但弱引用不会干扰它原本的释放
        _testCell = cell;
    }
    return cell;
}</span>

发现输入来不是空,且内存地址一致,说明这个被引用的cell时没有释放的。也就是说划出屏幕的cell是没有被释放的,但是cell里面的内容为什么更改的呢?是因为屏幕外的cell在进入屏幕内时,tableView会调用

<span style="font-size:18px;">-(UITableViewCell*)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath</span>

这个委托方法,然后我们会给它一个新的cell。就是tableview本身有一个,然后你又给了它一个,然后原来的就被挤掉了。所以这里的关键问题就变成了,在上面的这个委托方法里是否总是返回新构建的cell还是返回之前已有的cell的区别。

为什么以前没有注意过这个问题?我自身的理解过程来看,在开始接触tableView的时候,都是使用它来承载label。因为label是不可编辑的,所以显示内容必定是有一个固定的数据源,比如一个装满NSString的数组,然后每构建一个cell就丢给它一个字符串显示,这时就算是使用了新的cell那也没有问题,因为内容总是会被再次加上去。而textField是编辑的,一般初始状态是不给它text赋值的,这样构建一个新的cell的时候它的内容就是空的,那么看起来就像是已编辑的内容消失了。所以label的性质让我一开始并没有发现这些问题。

怎么让textField在划出屏幕后再回来内容还在呢?就是把原来的内容保存下来,再次调用

<span style="font-size:18px;">-(UITableViewCell*)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath</span>

的时候,把原来的东西丢回去。所以有3个层次:保存cell、保存cell上面的内容承载控件(UITextField、UIButton等),保存cell上面的内容(比如UITextField里面的文字)。具体操作就是:

(1)、保存cell:前面已经说了,cell其实没有被释放,是因为我们给了tableView一个新的cell替换掉了原来的,所以导致原来的内容没了。所以这里需要一个数组,把所有已经的cell保存进来,然后在每次需要cell的时候先从这个数组里面取,如果有就拿出来用。就像上面写的:

<span style="font-size:18px;"> if (indexPath.row == 1) {
        NSLog(@"_testCell >>> %@",_testCell);
        if (_testCell != nil) {<span style="white-space:pre">		</span>//如果不为空就直接返回这个cell
            return _testCell;
        }
    }</span>

当然如果是数组保存,那么还需要给每个cell添加一个标识,以便从数组里面筛选出正确的。发现照着这个思路,直接就变成了UITableView的重用机制了,而cell的ReuseIdentifier不就是用来筛选的标识吗?虽然不知道UITableView本身的源码是怎么处理的,但我看过有些源码,有类似的重用机制,也都是把东西放进一个数组,然后有需要的就取出来重新使用。那为什么要自己构建一个数组来保存cell而不是直接使用系统的重用呢?因为系统的重用是同一个ReuseIdentifier的cell是通用的,比如你有3个cell是同样的布局,假设叫A、B、C,然后都有一个UITextField来填写内容,那么如果它们使用同样的ReuseIdentifier,那么需要A的时候系统可能会把C给你,那么就会乱套。自己构建数组,自己筛选、自己设定标识,那么每个cell都有自己独特的标志,该是它的时候就是它,绝对不会重。当然,这样就必须把所有的cell保存下来,内容上是不是会有隐患?我的理解是,一般需要进行输入的tableView都不会是很长的,不可能会有一个界面,需要用户自己手动输入100行信息吧?!

还有一点,cell = [tableView
cellForRowAtIndexPath:indexPath] 这样是取不到屏幕外的cell的,文档里说了、也试了,即使那个cell其实根本没释放。

(2)保存承载内容的控件:这太累了,每次cell都建都要先判断这个控件是否存在,不存在构建,然后添加到cell上面,如果每个cell长得不一样,那么每个cell都得单独出来判断

(3)保存cell上的内容:就是在内容修改后,保存下来,然后在cell构建的时候,再把内容填回去。这个其实也挺累的,关键在于需要把内容实时的保存下来,比如textField,你必须要设置委托,在编辑结束时,及时的把他的内容存起来,否则下次textField构建,你给它什么东西呢?那假如这个tableview里还有UITextView,那又要设置UITextView的委托,还有button呢?假设每一种都有多个呢?还得区分开来,因为委托方法只有一个。

时间: 2024-11-05 11:34:29

UITableView的cell的释放、内容消失等的思考总结的相关文章

使用Autolayout实现UITableView的Cell动态布局和高度动态改变

本文翻译自:stackoverflow 有人在stackoverflow上问了一个问题: 如何在UITableViewCell中使用Autolayout来实现Cell的内容和子视图自动计算行高,并且能够保持平滑滚动的? 这个问题得到了300+的支持和450+的收藏,答案得到了730+的支持,很详细的说明了如何在iOS7和iOS8上实现UITableView的动态行高功能,并且这个答案对实现UICollectionView的动态行高也具有参考意义.所以在这里将这个答案翻译了一下,希望对大家有所帮助

iOS8+ UITableView自动计算cell高度并缓存

这篇文章我们来讲一下UITableView的cell自适应高度,以及遇到的问题的解决办法.在看文章之前希望你已经会UITableView的基本使用了. 先奉上这篇文章的demo的Github地址:UITableViewCellHeightDemo.大家可以下载下来和文章配合看. cell高度计算的历史 在iOS8之前,如果UITableViewCell的高度是动态的,如果想要显示正确的话,我们需要在下面这个UITableView的代理方法中,返回每一行的精确高度: - (CGFloat)tabl

UITableView设置cell为不可选?

本文选自StackOverflow(简称:SOF)精选问答汇总系列文章之一,本系列文章将为读者分享国外最优质的精彩问与答,供读者学习和了解国外最新技术.本文将为读者讲解UITableView如何设置单元格属性为不可选? 问题: Mugunth 怎样才能设置UITableView的单元格属性为不可选择.我不想看到用户选择单元格时显示蓝色选择区域. 答案: Daniel Dickison 把表格单元格selectionStyle的属性设置为UITableViewCellSelectionStyleN

iOS中TableViewController的cell重用出错(内容错乱)

iOS中TableViewController的cell重用出错(内容错乱) - 简书 时间 2016 用我的双手,成就你的梦想 ---栋哥 今天我的心爱的程序又出现bug,对于我这个小菜来说,不得不说是非常苦恼的,主要是cell加载的时候因为重用池的问题而出现各种的bug,虽然程序没有崩掉,但是大大影响到我的心情,下面是最主要的一个问题 ,就是cell的重用问题, cell因为从重用池中调取,没有及时删除上面的内容而导致内容的各种出现, 这里有几个解决方案. UITableView继承自UIS

使用Autolayout实现UITableView的Cell动态布局

如何在UITableViewCell中使用Autolayout来实现Cell的内容和子视图自动计算行高,并且能够保持平滑滚动的? 这个问题得到了300+的支持和450+的收藏,答案得到了730+的支持,很详细的说明了如何在iOS7和iOS8上实现UITableView的动态行高功能,并且这个答案对实现UICollectionView的动态行高也具有参考意义.所以在这里将这个答案翻译了一下,希望对大家有所帮助.以下是答案的全文翻译: 答案略长,如果你不喜欢细读,可以直接看这两个示例的代码: iOS

iOS如何固定UITableView中cell.imageView.image的图片大小

凡是进行ios开发的,基本上都会遇到要展示列表,或者即使不是标准列表,但由于数量不固定,也需要如同列表一样从上往下显示.加载的情况.这些,都绕不过对UITableView的使用. 在iOS开发中UITableView可以说是使用最广泛的控件,我们平时使用的软件中到处都可以看到它的影子,类似于微信.QQ.新浪微博等软件基本上随处都是UITableView.当然它的广泛使用自然离不开它强大的功能. 我们经常在开发过程中会用到默认UITableView的cell.imageView.image,如果图

[IOS开发教程] iOS如何固定UITableView中cell.imageView.image的图片大小

凡是进行ios开发的,基本上都会遇到要展示列表,或者即使不是标准列表,但由于数量不固定,也需要如同列表一样从上往下显示.加载的情况.这些,都绕不过对UITableView的使用. 在iOS开发中UITableView可以说是使用最广泛的控件,我们平时使用的软件中到处都可以看到它的影子,类似于微信.QQ.新浪微博等软件基本上随处都是UITableView.当然它的广泛使用自然离不开它强大的功能. 我们经常在开发过程中会用到默认UITableView的cell.imageView.image,如果图

视图交互--表视图(UITableView)的cell交互析略

在表视图UITableView的cell上经常有一些交互,根据项目开发中的情况,需要对此进行一些规范.总结出了几种交互方法,这些方法在其他视图的交互上同样可以适用.用一个简单的例子来举例说明一下,其他交互类型与此类似,希望读者举一反三. 需求:在一个cell上,当点击按钮时,将左侧的文字同样展示在右侧显示部分.    这个需求,最主要的部分,就是获取所点击按钮的cell的问题,然后通过cell将左侧的label上的文字显示在右侧的label上. 方法: (一)交互第一种:给cell的按钮tag赋

iOS项目开发小技巧 (三) --UITableView实现Cell左划删除等自定义功能

今天来介绍下iOS开发中UITableView的Cell左划实现微信中置顶,删除等功能.该功能在iOS8.0以前是需要很复杂的实现,不过github上应该有现成demo,不过今天介绍的是在iOS8.0以后苹果新推出的api,来实现Cell左划自定义控件. 1. 首先创建UITableView视图,实现其俩个代理,UITableViewDelegate和UITableViewDataSource,该处代码就不说了,主要是俩个回调方法 -(UITableViewCell *)tableView:(U