注册
iOS

iOS UITableView 图片刷新闪烁问题记录

一. 问题背景


项目中遇到一个问题,就是当App不在首页的时候,切换到其他App比如微信,然后返回App当前页面,然后从当前页面返回首页,会在首页viewWillAppear这里去拉取是否有未完成订单的接口,刷新UITableView,这时会出现广告位闪烁问题。


8f63155dda18fd13a22637c6db5026aa.gif


二. 问题排查


1.原因分析


这个问题经过断点调试和排除法,发现只要当App进入后台后,回来刷新首页的UITableView都有可能出现闪烁现象。


因此首先我们对图片的加载做延迟操作,并在Cell生成方法调用里面添加相关打印:


1ad655213f82ee16bf83d8ce3515c9af.png


d15873793308184b5c848c984be6e1c0.png


可以看到如下打印日志:


2c69f15f1b3f040a11a33ccc4b154636.png


从打印日志我们可以看出来,调用reloadData方法后,原来UITableViewcell位置会调整。


但是如果我们App没有进入后台,而是直接调用UITableViewreloadData方法,并不会出现闪烁现象。


因此可以这里可以推测应该是进入后台做了什么操作导致,回到App刷新才会导致闪烁。


因为使用的是SDWebImage加载框架加载,我们合理的怀疑是加载图片的SDWebImage框架,进入后台的处理逻辑导致的,因此我们先使用imageCacheDict字典写下图片加载和缓存逻辑:


9699584d1187a69dda372b6b1edc38ca.png


经测试,进入后台,再返回App刷新不会出现闪烁现象。


因此可以肯定UITableView调用reloadData方法闪烁原因是SDWebImage,在进入后台的时候对内存缓存做了相关操作导致。


我们都知道SDWebImage,默认是使用NSCache来做内存缓存,而NSCache在进入后台的时候,默认会清空缓存操作,导致返回App调用UITableView调用reloadData方法时候,SDWebImage需要根据图片地址重新去磁盘获取图像数据,然后解压解码渲染,因为是从缓存磁盘直接获取图像数据,没有渲染流程,因此会造成闪烁。


为了验证这个猜想,我们使用YYWebImage加载框架来做对比实验:

首先注释掉YYWebImage进入后台清空内存缓存的逻辑: 

209147cb970bf40deb2dbd7cbfd4accf.png

然后进入后台,返回App调用UITableView调用reloadData刷新,发现一切正常。

原因总结:

  • 第一个原因是UITableView调用reloadData方法,由于UITableViewCell的复用,会出现Cell位置调整现象

  • 由于SDWebImage使用了NSCache做内存缓存,当App进入后台,NSCache会清空内存缓存,导致返回App后调用UITableView调用reloadData,刷新去加载图片的时候,需要从SDWebImage的磁盘中重新获取图片数据,然后重新解压解码渲染,因为从磁盘中读取速度快,两者原因导致了闪烁。


三. 解决方案


因为该现象是由如上两个原因导致,因此针对这两个原因,有如下两种解决方案:

1. 解决UITableViewCell复用问题


可以通过设置ReusableCellWithIdentifier不同,保证广告cell不进行复用。

 NSString *cellId = [NSString stringWithFormat:@"%ld-%ld-FJFAdTableViewCell", indexPath.section, indexPath.row];

2. 从后台返回后,提早进行刷新操作

当从后台返回App前台的时候或者视图添加到父视图的时候,先执行下UITableView调用reloadData方法,提前通过SDWebImage去从磁盘中加载图片。


从后台返回前台:

[NSNotificationCenter.defaultCenter addObserver:self selector:@selector(willEnterForeground) name:UIApplicationWillEnterForegroundNotification object:nil];
- (void)willEnterForeground {
[self.tableView reloadData];
NSLog(@"--------------------------willEnterForeground");
}

视图添加到父视图:

- (void)willMoveToParentViewController:(UIViewController *)parent {
[self.tableView reloadData];
}

作者:果哥爸
链接:https://juejin.cn/post/7267091810367111224
来源:稀土掘金
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

0 个评论

要回复文章请先登录注册