iOS一直讲的单元格优化

1 复用Cell(dequeueReusableCellWithIdentifier:);

2 异步解码图片、离屏渲染(如使用CATiledLayer);

3减少AutoLayout计算,预计算Cell高度;

4 使用willDisplayCell延迟加载非关键资源

异步解码图片、离屏渲染(如使用CATiledLayer)这两个之前没有开发过,今天写一个demo记录一下

swift 复制代码
/*
 异步解码图片,避免单元格卡顿
 当你用 UIImage(named:) 或 UIImage(data:) 加载图片时,系统会延迟解码(lazy decode):
     •    图片只在真正显示到屏幕上时才会解码。
     •    解码操作发生在主线程上 → 容易造成 UI 卡顿(掉帧)。
 */
class DecodeImages {
    //可以结合 DispatchSemaphore 控制并发,避免同时解码过多大图导致内存暴涨
    //信号量限制最大解码数量为2
    let semaphore = DispatchSemaphore(value: 2)
    let queue = DispatchQueue(label: "www.baidksk.com",attributes: DispatchQueue.Attributes.concurrent)
    func asyncDecodeImage(_ data: Data, complete: @escaping (UIImage?)->Void) {
        semaphore.wait()
        defer {semaphore.signal()}
        guard let imageSource = CGImageSourceCreateWithData(data as CFData, nil),
              let cgImage = CGImageSourceCreateImageAtIndex(imageSource, 0, nil) else {
            DispatchQueue.main.async {
                complete(nil)
            }
            return
        }
        let decodedImage = UIImage(cgImage: cgImage,scale: UIScreen.main.scale,orientation: .up)
        DispatchQueue.main.async {
            complete(decodedImage)
        }
    }
    
    
    
    
}
/*
 当一张超大图(如地图、PDF、超高清图)显示在 UIScrollView 或 CALayer 上时,
 如果一次性全部解码、渲染,会非常消耗内存。
 
 解决思路
 使用 CATiledLayer 分块(tile)绘制:
 系统只绘制当前可见区域的部分图块,每块独立解码与渲染。
 
 •    CATiledLayer 自动多线程渲染;
 •    每个 tile 的渲染在后台进行;
 •    iOS 系统内部使用 GCD 调度线程;
 •    你可以通过 DispatchSemaphore 控制最大渲染任务数量,以防 GPU/CPU 占用过高。
 */

class OutScreenRenderView : UIView {
    private let image: CGImage
    override class var layerClass: AnyClass {
        CATiledLayer.self
    }
    init(image: CGImage) {
        self.image = image
        super.init(frame: .zero)
        
        let titleLayer = self.layer as! CATiledLayer
        titleLayer.levelsOfDetail = 4
        titleLayer.tileSize = CGSize(width: 256, height: 256)
        contentScaleFactor = 1.0
    }
    
    required init?(coder: NSCoder) {
        fatalError("init(coder:) has not been implemented")
    }
    
    override func draw(_ layer: CALayer, in ctx: CGContext) {
        let rect = ctx.boundingBoxOfClipPath
        let scale = layer.contentsScale
        let imageRect = CGRect(x: 0, y: 0, width: CGFloat(image.width), height: CGFloat(image.height))
        ctx.saveGState()
        ctx.translateBy(x: 0, y: imageRect.size.height)
        ctx.draw(image, in: rect)
        ctx.restoreGState()
    }
}
相关推荐
tiantian_cool8 小时前
Claude Code 四大核心技能使用指南
ios
冰淇淋真好吃12 小时前
iOS实现 WKWebView 长截图的优雅方案
ios
栗子叶12 小时前
IP协议 地址划分&MAC地址作用&ip addr命令
网络·tcp/ip·macos
draking16 小时前
Anthropic 封杀当天,我把 OpenCode 升到 1.1.11,踩了 5 个坑
macos·ai编程
Roye_ack17 小时前
【Mac环境配置教程】深度学习环境配置(Anaconda + PyTorch)
macos
前端不太难17 小时前
Flutter / RN / iOS,在长期维护下的性能差异本质
flutter·ios
程序员雄杰17 小时前
腾讯云轻量应用服务器mac中ssh免密登录到服务器
macos·ssh·腾讯云
YongPagani1 天前
Mac安装Homebrew
macos
Byron Loong2 天前
【系统】Mac系统和Linux 指令对比
linux·macos·策略模式
软件小滔2 天前
拖拽出来的专业感
经验分享·macos·mac·应用推荐