Swift 中懒加载相关内容

1. lazy

1
2
3
4
5
6
7
8
9
class NetworkManager {
lazy var urlSession: URLSession = {
let configuration = URLSessionConfiguration.default
let urlSession = URLSession(configuration: configuration)
return urlSession
}()

// ...
}

lazy 为 Swift 中的关键字,可以用于类和结构体的存储变量属性,表示该属性只在首次进行访问时初始值才会计算和存储。

需要注意的是:

If a property marked with the lazy modifier is accessed by multiple threads simultaneously and the property has not yet been initialized, there is no guarantee that the property will be initialized only once.
如果标记有 lazy 修饰符的属性同时被多个线程访问,并且该属性尚未初始化,则不能保证该属性仅被初始化一次。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
import Foundation

class LazyPropertyClass {
lazy var value: Int = {
return Int.random(in: 0...100)
}()
}

let instance = LazyPropertyClass()

for i in 0...3 {
DispatchQueue(label: "\(i)", attributes: .concurrent).async {
print(instance.value)
}
}
// 47
// 53
// 67
// 67

2. 全局属性

1
2
3
4
struct ViewConstants {
static let padding: CGFloat = 12.0
static let toolbarHeight: CGFloat = 56.0
}

全局变量是定义在任何函数、方法、闭包之外或者其他类型上下文的变量。
局部变量是定义在函数、方法之内或者闭包上下文的变量

https://docs.swift.org/swift-book/LanguageGuide/Properties.html#ID263 有下面一段话。

Global constants and variables are always computed lazily, in a similar manner to Lazy Stored Properties. Unlike lazy stored properties, global constants and variables do not need to be marked with the lazy modifier.
全局常量和变量总是延迟计算,与惰性存储属性类似的方式。与惰性存储的属性不同的是,全局常量和变量不需要使用 lazy 修饰符进行标记。

Local constants and variables are never computed lazily.
局部常量和变量永远不会延迟计算。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
import Foundation

let value1: Int = {
print("load value1")
return Int.random(in: 0...100)
}()

class GlobalAndLocalPropertyClass {

static let value2: Int = {
print("load value2")
return Int.random(in: 0...100)
}()

static var value3: Int = {
print("load value3")
return Int.random(in: 0...100)
}()

lazy var value4: Int = {
print("load value4")
return Int.random(in: 0...100)
}()

let value5: Int = {
print("load value5")
return Int.random(in: 0...100)
}()

var value6: Int = {
print("load value6")
return Int.random(in: 0...100)
}()
}

// 初始化 GlobalAndLocalPropertyClass
// _ = GlobalAndLocalPropertyClass()

// 运行打印:
// load value5
// load value6

注意,在 Playground 中执行:

1
2
3
// load value1
// load value5
// load value6

3. LazySequenceProtocol

1
2
let doubled = [1, 2, 3].lazy.map { $0 * 2 }
print(doubled)
  • LazyCollectionProtocol 继承 LazySequenceProtocol协议
  • Lazy 表示延迟执行
  • Sequence: 一种类型,提供对元素的顺序和迭代访问
  • Collection: 一个序列,元素可以无损地遍历多次,并由索引下标访问
  • 避免不必要的存储分配和计算。

符合 LazySequenceProtocol 的类型

  • LazySequence
  • LazyMapSequence
  • LazyDropWhileSequence
  • LazyFilterSequence
  • LazyPrefixWhileSequence
  • ReversedCollection

当且仅当 Base 符合 LazySequenceProtocol 时才符合

  • Slice

当且仅当 Base 符合 LazySequenceProtocol 时才符合

  • LazyCollection
1
typealias LazyCollection<T> = LazySequence<T> where T : Collection
  • LazyMapCollection
1
typealias LazyMapCollection<T, U> = LazyMapSequence<T, U> where T : Collection
  • LazyDropWhileCollection
1
typealias LazyDropWhileCollection<T> = LazyDropWhileSequence<T> where T : Collection
  • LazyFilterCollection
1
typealias LazyFilterCollection<T> = LazyFilterSequence<T> where T : Collection
  • LazyPrefixWhileCollection
1
typealias LazyPrefixWhileCollection<T> = LazyPrefixWhileSequence<T> where T : Collection

使用

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
// Array 有一个lazy属性返回类型为LazySequence<Array<Element>>
let arr = [1, 2 ,3]

let ls = arr.lazy
print(type(of: ls)
// LazySequence<Array<Int>>

let doubled = ls.map { $0* 2 }
print(type(of: doubled))
// LazyMapSequence<Array<Int>, Int>

let greaterThanOne = ls.filter { $0 > 1 }
print(type(of: greaterThanOne))
// LazyFilterSequence<Array<Int>>

let lessThanTwo = ls.lazy.drop { $0 < 2 }
print(type(of: lessThanTwo))
// LazyDropWhileSequence<Array<Int>>

let lessonThanThree = ls.lazy.prefix { $0 < 3 }
// LazyPrefixWhileSequence<Array<Int>>

参考