UITableView 卡顿排查记实

前言

项目中实现了一个歌曲列表,Cell 的内容是歌曲封面和标题的简单内容,但是在滑动时会出现卡顿,本文记录下排查过程。

使用 Instruments

对于卡顿掉帧问题大多数都是在主线程进行了较大的耗时操作,所以使用 Instruments 中 Time Profile 来检测下具体是什么方法耗时较长。

可以看到在 Runloop 的回调中执行了 CA::Transaction::commit(),之后调用了 ImageIO 框架进行图片的解码。那么问题来了,为什么加载其他图片不会?这里执行解码的类是 PNGReadPlugin,会不会是 PNG 图片解码就会卡顿?

查看源码

objc 复制代码
+ (BOOL)shouldDecodeImage:(nullable UIImage *)image {
    // Prevent "CGBitmapContextCreateImage: invalid context 0x0" error
    if (image == nil) {
        return NO;
    }
    
    // do not decode animated images
    if (image.images != nil) {
        return NO;
    }
    
    CGImageRef imageRef = image.CGImage;
    
    BOOL hasAlpha = SDCGImageRefContainsAlpha(imageRef);
    // do not decode images with alpha
    if (hasAlpha) {
        return NO;
    }
    
    return YES;
}

经过一番查找,找到了这段关键代码。可以看到,这里判断了图片是否包含 Alpha 通道,如果包含则不进行解码。所以刚才的猜测是正确的,PNG 图片会包含 Alpha 通道,SDWebImage 不会对其进行解码,造成了卡顿。

之前对于文章《iOS 处理图片的一些小 Tip》中提到的"UIImage 第一次显示到屏幕时才会被解码"不太理解,这次算是遇上了。

解决方案

升级 SDWebImage

由于一些历史原因,项目中使用的 SDWebImage 版本还是比较老的 4.2.3 版本,而最新版本的 SDWebImage 已经支持对包含 alpha 通道的 PNG 图片进行解码

objc 复制代码
+ (BOOL)shouldDecodeImage:(nullable UIImage *)image {
    // Avoid extra decode
    if (image.sd_isDecoded) {
        return NO;
    }
    // Prevent "CGBitmapContextCreateImage: invalid context 0x0" error
    if (image == nil) {
        return NO;
    }
    // do not decode animated images
    if (image.images != nil) {
        return NO;
    }

    return YES;
}

服务端裁剪成小图

造成卡顿的另一个原因是直接使用了原图,分辨率较大,可以通过在图片 URL 上拼接相关参数获取一张小图,这个方案有诸多优点:

  • 显示效果更佳
  • 节省用户流量
  • 提高加载速度

延伸思考

NSData 转化成 UIImage 不是就完成解码了吗?

SDWebImage 的解码做了什么,和系统 ImageIO 的解码有什么不同?

系统为什么要把图片解码这么耗时的操作放到主线程?

带着这些疑问找到 SDWebImage 的主要维护作者的文章 主流图片加载库所使用的预解码究竟干了什么,解答了我的所有疑惑。

这里简单总结一下:

  • NSData 转化成 UIImage 之后,它的 Bitmap 是没有立即创建的,这个 UIImage 只是包含一些元信息的空壳,等到最终显示到屏幕上才会创建。
  • SDWebImage 等第三方库通过在子线程调用 CGContextDrawImage让 ImageIO 立即解码分配 Bitmap 内存。
  • iOS 早期设备的内存非常有限,所有图片都提前进行解码会增加内存的压力。
相关推荐
Frank学习路上30 分钟前
【IOS】XCode创建firstapp并运行(成为IOS开发者)
开发语言·学习·ios·cocoa·xcode
瓜子三百克7 小时前
CALayer的异步处理
macos·ios·cocoa
吴Wu涛涛涛涛涛Tao8 小时前
一步到位:用 Very Good CLI × Bloc × go_router 打好 Flutter 工程地基
flutter·ios
杂雾无尘10 小时前
开发者必看:如何在 iOS 应用中完美实现动态自定义字体!
ios·swift·apple
kymjs张涛12 小时前
零一开源|前沿技术周报 #6
前端·ios·harmonyos
与火星的孩子对话1 天前
Unity进阶课程【六】Android、ios、Pad 终端设备打包局域网IP调试、USB调试、性能检测、控制台打印日志等、C#
android·unity·ios·c#·ip
恋猫de小郭2 天前
Flutter Widget Preview 功能已合并到 master,提前在体验毛坯的预览支持
android·flutter·ios
点金石游戏出海2 天前
每周资讯 | Krafton斥资750亿日元收购日本动画公司ADK;《崩坏:星穹铁道》新版本首日登顶iOS畅销榜
游戏·ios·业界资讯·apple·崩坏星穹铁道
旷世奇才李先生2 天前
Swift 安装使用教程
开发语言·ios·swift
90后的晨仔2 天前
Xcode16报错: SDK does not contain 'libarclite' at the path '/Applicati
ios