TableViewCell重用机制
UITableView继承自UIScrollview,是苹果为我们封装好的一个基于scroll的控件。
UITableView中的cell可以有很多,一般会通过重用cell来达到节省内存的目的:并不为每个数据创建一个UITableViewCell,我们只创建屏幕可显示的最大的cell个数+1,然后去循环重复使用这些cell(通过为每个cell指定一个重用标识符(reuseIdentifier),即指定了单元格的种类。当cell滚出屏幕时,会将滚出屏幕的单元格放入重用的queue中,当某个未在屏幕上的单元格要显示的时候,就从这个queue中取出单元格进行重用)。
// 设置单元格 indexPath :单元格当前所在位置 -- 哪个分区哪一行等
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath //UITableViewDataSource
{
static NSString *identifier = @"cell" ;
//相当于从集合中找寻完全出屏幕的单元格.
// identifier : 因为一个表视图中可能存在多种样式的单元格,咱们把相同样式的单元格放到同一个集合里面,为这个集合加标示符,当我们需要用到某种样式的单元格的时候,根据不同的标示符,从不同的集合中找寻单元格.
UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:identifier] ;
// 如果从集合中未找到单元格,也就是集合中还没有单元格,也就是还没有单元格出屏幕,那么我们就需要创建单元格
if (!cell)
{
// 创建cell的时候需要标示符(Identifier)是因为,当该cell出屏幕的时候需要根据标示符放到对应的集合中.
cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleSubtitle reuseIdentifier:@"cell"] ;
return cell ;
}
TableViewCell重用机制避免重复显示的方法
但对于多变的自定义cell,有时这种重用机制会出错。比如,当一个cell含有一个UIView的子类并被放在重用queue中以待重用,这时如果一个未包含任何子视图的cell要显示在屏幕上,就会取出并使用这个重用的cell显示在无任何子视图的cell中,这时候就会出错。这时候需要对取出的重用的cell做重新赋值,从而解决cell的重用机制导致cell的内容重复显示的问题。
解决方案1:取消cell的重用机制,通过indexPath来创建cell.显示数据大时内存占用较多。🌰:
//解决方案1:取消cell的重用机制,通过indexPath来创建cell.显示数据大时内存占用较多
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
{
if (tableView == self.classicRouteTableView) {
//通过indexPath创建cell
ClassicRouteTableViewCell *cell = [tableView cellForRowAtIndexPath:indexPath];
// 如果cell不存在就初始化cell
if (!cell) {
cell = [[ClassicRouteTableViewCell alloc]initWithStyle:UITableViewCellStyleDefault reuseIdentifier:classicRouteCell];
}
cell.selectionStyle = UITableViewCellSelectionStyleNone;
if (self.classicRouteDataArray.count != 0) {
NSMutableArray *EachRoute =[NSMutableArray arrayWithArray:[self.classicRouteDataArray[indexPath.section] EachRoute]];
cell.EachRoute = EachRoute;
}
return cell;
}
if (tableView == self.hotRouteTableView) {
//通过indexPath创建cell
HotRouteTableViewCell *cell = [tableView cellForRowAtIndexPath:indexPath];
// 如果cell不存在就初始化cell
if (!cell) {
cell = [[HotRouteTableViewCell alloc]initWithStyle:UITableViewCellStyleDefault reuseIdentifier:hotRouteCell];
}
cell.selectionStyle = UITableViewCellSelectionStyleNone;
if (self.hotRouteDataArray.count != 0) {
NSMutableArray *EachRoute =[NSMutableArray arrayWithArray:[self.hotRouteDataArray[indexPath.section] EachRoute]];
cell.EachRoute = EachRoute;
}
return cell;
}
return nil;
}
解决方案2:让每个cell都拥有一个对应的标识.显示数据大时内存占用较多。🌰:
//解决方案2:让每个cell都拥有一个对应的标识.显示数据大时内存占用较多
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
{
if (tableView == self.classicRouteTableView) {
// 每次先从字典中根据IndexPath取出唯一标识符
NSString *identifier = [self.identifierDic objectForKey:[NSString stringWithFormat:@"classicRouteCell%@",indexPath]];
// 如果取出的唯一标示符不存在,则初始化唯一标示符,并将其存入字典中,对应唯一标示符注册Cell
if (identifier == nil) {
identifier = [NSString stringWithFormat:@"%@", indexPath];
[self.identifierDic setValue:identifier forKey:[NSString stringWithFormat:@"classicRouteCell%@",indexPath]];
// 注册Cell
[self.classicRouteTableView registerClass:[ClassicRouteTableViewCell class] forCellReuseIdentifier:identifier];
}
ClassicRouteTableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:identifier forIndexPath:indexPath];
cell.selectionStyle = UITableViewCellSelectionStyleNone;
if (self.classicRouteDataArray.count != 0) {
NSMutableArray *EachRoute =[NSMutableArray arrayWithArray:[self.classicRouteDataArray[indexPath.section] EachRoute]];
cell.EachRoute = EachRoute;
}
return cell;
}
if (tableView == self.hotRouteTableView) {
// 每次先从字典中根据IndexPath取出唯一标识符
NSString *identifier1 = [self.identifierDic objectForKey:[NSString stringWithFormat:@"hotRouteCell%ld%@",self.hotDayCount, indexPath]];
// 如果取出的唯一标示符不存在,则初始化唯一标示符,并将其存入字典中,对应唯一标示符注册Cell
if (identifier1 == nil) {
identifier1 = [NSString stringWithFormat:@"%@", indexPath];
[self.identifierDic setValue:identifier1 forKey:[NSString stringWithFormat:@"hotRouteCell%ld%@",self.hotDayCount, indexPath]];
// 注册Cell
[self.hotRouteTableView registerClass:[HotRouteTableViewCell class] forCellReuseIdentifier:identifier1];
}
HotRouteTableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:identifier1 forIndexPath:indexPath];
cell.selectionStyle = UITableViewCellSelectionStyleNone;
if (self.hotRouteDataArray.count != 0) {
NSMutableArray *EachRoute =[NSMutableArray arrayWithArray:[self.hotRouteDataArray[indexPath.section] EachRoute]];
cell.EachRoute = EachRoute;
}
return cell;
}
return nil;
}
解决方案3:只要最后一个显示的cell内容不为空,将其子视图删除,成为一个空白的cell,再进行自定义。🌰:
//解决方案3:只要最后一个显示的cell内容不为空,将其子视图删除,成为一个空白的cell,再进行自定义
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
{
if (tableView == self.classicRouteTableView) {
// 通过唯一标识符classicRouteCell创建cell实例
ClassicRouteTableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:classicRouteCell forIndexPath:indexPath];
//当cell存在且最后一个存在时 将最后的一个cell删除其子视图,成为一个空白的cell,再进行自定义cell
//这里是使用while语句 或if语句
if (cell) {
while ([cell.contentView.subviews lastObject] != nil) {
[[cell.contentView.subviews lastObject] removeFromSuperview];
}
}
cell.selectionStyle = UITableViewCellSelectionStyleNone;
if (self.classicRouteDataArray.count != 0) {
NSMutableArray *EachRoute =[NSMutableArray arrayWithArray:[self.classicRouteDataArray[indexPath.section] EachRoute]];
cell.EachRoute = EachRoute;
}
return cell;
}
if (tableView == self.hotRouteTableView) {
// 通过唯一标识符hotRouteCell创建cell实例
HotRouteTableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:hotRouteCell forIndexPath:indexPath];
//当cell存在且最后一个存在时 将最后的一个cell删除其子视图,成为一个空白的cell,再进行自定义cell
//这里是使用while语句 或if语句
if (cell) {
if ([cell.contentView.subviews lastObject] != nil) {
[[cell.contentView.subviews lastObject] removeFromSuperview];
}
}
cell.selectionStyle = UITableViewCellSelectionStyleNone;
if (self.hotRouteDataArray.count != 0) {
NSMutableArray *EachRoute =[NSMutableArray arrayWithArray:[self.hotRouteDataArray[indexPath.section] EachRoute]];
cell.EachRoute = EachRoute;
}
return cell;
}
return nil;
}
总结
-
当cell少或数据量不大时,可以添加标识符,或通过indexPath来创建cell的方法来解决cell的重用问题。
-
当cell重用较多时或数据量大时,删除cell的子视图的方法来解决cell的常用问题。