如何在 Swift 中使用 @autoclosure 来提高性能

关注我,每天分享一个关于 iOS 的新知识

前言

在 Swift,有一种不常用的闭包类型 -- @autoclosure这个关键字可以让我们将一个表达式封装到一个闭包中,在需要该表达式结果的时候才执行该闭包(类似于懒加载)。这为我们提供了一种延迟执行代码的方式,可以有效提高性能。

虽然业务开发中不经常使用,但是在 swift 标准库中使用场景非常多,关于怎么使用我们来用一些示例来详细聊聊。

使用方法

举个例子,在此示例中,我们创建了一个 log 方法和一个 Person 结构体,当调用 description 属性时,将会打印日志:

swift 复制代码
struct Person {
    let name: String
    
    var description: String {
        print("调用了 Person description.")
        return "Person name is \(name)"
    }
}

func log(_ message: String) {
    if false {
        print("\(message)")
    }
}

let person = Person(name: "iOS 新知")
log(person.description)

执行上边的代码,将会看到控制台输出"调用了 Person description."

因为我的代码中写了 if false,所以永远不可能执行 if 条件下的打印,也就不会访问 message 这个参数,从代码效率上来说这时候 Persondescription 属性执行就是无效的(如果 description 中有大量的运算,会导致性能下降)。这是因为调用 person.description 的时候就直接执行了 description 这个计算属性。

让我们改一下代码,把 message 参数改成一个闭包,只有函数内调用这个闭包才执行 description 函数:

swift 复制代码
struct Person {
    let name: String
    
    var description: String {
        print("调用了 Person description.")
        return "Person name is \(name)"
    }
}

func log(_ message: () -> String) {
    if false {
        print("\(message())")
    }
}

let person = Person(name: "iOS 新知")
log({ person.description })

我把 log 函数的 message 参数由 String 类型,改成了 () -> String 闭包类型,调用的时候也改成了 log({ person.description }),此时再运行代码,没有执行 description 属性,符合我们的预期。

但是每次调用 log 函数都要将参数用大括号包裹,用起来十分的不便,这时候 @autoclosure 就要登场了 ,在 log 函数的 message 参数中增加这个关键字,调用的时候就不需要大括号了,最终代码如下:

swift 复制代码
struct Person {
    let name: String
    
    var description: String {
        print("调用了 Person description.")
        return "Person name is \(name)"
    }
}

func log(_ message: @autoclosure () -> String) {
    if false {
        print("\(message())")
    }
}

let person = Person(name: "iOS 新知")
log(person.description)

这份代码看起来就清爽多了,既实现了我们的需求(访问 message 的时候才调用 description),调用方式又和之前一样。

一些其他例子

1、延迟实例化对象

less 复制代码
struct Canvas {
    let width: CGFloat
    let height: CGFloat
    
    func drawShape() {}
}

func draw(canvas: @autoclosure () -> Canvas) {
    if shouldDraw {
        canvas().drawShape()
    }
}

// 调用时才创建 Canvas 实例
draw(canvas: Canvas(width: 200, height: 100))

Canvas 对象只在需要绘制时才创建调用 init 方法创建。

2、减少内存使用

less 复制代码
func decode(image: @autoclosure () -> UIImage) -> Image? {
    if shouldDecode {
        return decodeImage(image())
    } else {
        return nil
    }
}
let img = decode(image: loadImageFromDisk())

图片解码可能很占内存,应该只在需要时(shouldDecode)解码图片,避免不必要的内存占用。

3、避免冗余计算

less 复制代码
func hash(value: @autoclosure () -> String) -> Int {
    if let hashValue = self.hashValue {
        return hashValue
    }
    return hashFunction(value())
}
let hash1 = hash(value: someString)
let hash2 = hash(value: someString)

有些操作开销比较大,可以使用 @autoclosure 避免重复计算。

4、字典取值时的默认值计算

之前的文章中介绍过字典取值时可以设置默认值,其实在 swift 底层就用到了 @autoclosure

less 复制代码
public subscript(key: Key, default defaultValue: @autoclosure () -> Value) -> Value

因为在字典取不到 key 对应的值的情况下,才需要计算默认值,而不是每次都计算。

这里每天分享一个 iOS 的新知识,快来关注我吧

本文同步自微信公众号 "iOS新知",每天准时分享一个新知识,这里只是同步,想要及时学到就来关注我吧!

相关推荐
wjm0410067 小时前
ios八股文 -- Objective-c
开发语言·ios·objective-c
麦兜*16 小时前
Swift + Xcode 开发环境搭建终极指南
开发语言·ios·swiftui·xcode·swift·苹果vision pro·swift5.6.3
Digitally21 小时前
重置iPhone会删除所有内容吗? 详细回答
ios·iphone
普罗米拉稀1 天前
Flutter 复用艺术:Mixin 与 Abstract 的架构哲学与线性化解密
flutter·ios·面试
kymjs张涛1 天前
零一开源|前沿技术周刊 #12
ios·google·github
2501_915918412 天前
iOS 应用上架全流程实践,从开发内测到正式发布的多工具组合方案
android·ios·小程序·https·uni-app·iphone·webview
笔沫拾光2 天前
iOS 正式包签名指南
flutter·ios·ios签名
HarderCoder2 天前
Swift Concurrency:彻底告别“线程思维”,拥抱 Task 的世界
swift
HarderCoder2 天前
深入理解 Swift 中的 async/await:告别回调地狱,拥抱结构化并发
swift
Magnetic_h2 天前
【iOS】锁的原理
笔记·学习·macos·ios·objective-c·cocoa·xcode