关于UITableView中cell的重用问题
一、代码
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath { static NSString *cellIdentifier = @"CellIdentifier"; UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:cellIdentifier]; if (cell == nil) { cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:cellIdentifier]; } cell.textLabel.text = [[_data objectAtIndex:indexPath.row] objectForKey:@"name"]; return cell; }
二、代码测试情况如下:
1、在attributes inspector中设置过cell的Identifier,那么不会进入if(cell == nil)条件语句中。
2、没有设置cell的Identifier:
如果在3.5英寸设备中,cell==nil条件成立为12次,屏幕满屏显示11条;
如果在4英寸设备中,cell==nil条件成立为14次,屏幕满屏显示13条;
如果在4.7英寸设备中,cell==nil条件成立为16次,屏幕满屏显示15条;
如果在5.5英寸设备中,cell==nil条件成立为17次,屏幕满屏显示16条;
3、代码启动后,循环指定次数调用tableView:cellForRowAtIndexPath:方法,滑动时,最上/最下面cell移除屏幕时,再调用一次该方法。
三、重用机制原理
我们在UITableView头文件,看到可读的NSArray的visiableCells属性和dequeueResuableCellWithIdentifier:方法。
其实可以这样理解,内部包括私有两个属性NSMutableArray* visiableCells和NSMutableDictnery* reusableTableCells两个结构。visiableCells内保存当前显示的cells,reusableTableCells保存可重用的cells。
TableView显示之初,reusableTableCells为空,那么tableView dequeueReusableCellWithIdentifier:CellIdentifier返回nil。开始的cell都是通过[[UITableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:CellIdentifier]来创建,而且cellForRowAtIndexPath只是重复调用最大显示cell数的次数。
eg.有100条数据,iPhone一屏最多显示10个cell。程序最开始显示TableView的情况是:(过程已由苹果UITableView实现)
1. 用[[UITableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:CellIdentifier]创建10次cell,并给cell指定同样的重用标识(当然,可以为不同显示类型的cell指定不同的标识)。并且10个cell全部都加入到visiableCells数组,reusableTableCells为空。
2. 向下拖动tableView,当cell1完全移出屏幕,并且cell11(它也是alloc出来的,原因同上)完全显示出来的时候。cell11加入到visiableCells,cell1移出visiableCells,cell1加入到reusableTableCells。
3. 接着向下拖动tableView,因为reusableTableCells中已经有值,所以,当需要显示新的cell,cellForRowAtIndexPath再次被调用的时候,tableView dequeueReusableCellWithIdentifier:CellIdentifier,返回cell1。cell1加入到visiableCells,cell1移出reusableTableCells;cell2移出visiableCells,cell2加入到reusableTableCells。之后再需要显示的Cell就可以正常重用了。
所以整个过程并不难理解,但需要注意正是因为这样的原因:配置Cell的时候一定要注意,对取出的重用的cell做重新赋值,不要遗留老数据。