Swift 5.6 新特性

Swift 5.6 新特性

不可用条件(#unavailable)

#available 用于根据不同的平台、版本进行条件编译:

  1. if #available(iOS 15, *) {
  2. // 通配符 * 表示 Apple 所有平台,如果是 iOS,则要求 >= 15
  3. } else {
  4. // iOS 15 以下
  5. }

Swift 5.6 引入了 #unavailable ,它和 #available 的意思正好相反,下面的示例和上面示例中的 else 分支表达的意思是一样的。

  1. if #unavailable(iOS 15) {
  2. // iOS 15 以下
  3. }

它也支持同时指定多个平台:

  1. if #unavailable(iOS 15, macOS 12) {
  2. // iOS 15 以下, macOS 12 以下
  3. }

类型占位符(?)

Swift 5.6 支持使用占位符 __? 表示需要声明的类型,我们无需显示地指定类型,编译器会根据上下文自行推断。

  1. let complexType: [Int: _] = [1: [1], 2: [[2]], 3: [(1, 2)], 4: [{}]]
  2. // 编译器会将 complexType 的类型推断为 [Int : [Any]]
  3. let complexType: [Int: _] = [1: [1], 2: [[2]], 3: [(1, 2)], 4: nil]
  4. // 编译器会将 complexType 的类型推断为 [Int : [Any]?]

CodingKeyRepresentable 协议

先来看看下面的代码,我们想对字典进行编码,这个字典有点特殊,它的 key 是枚举类型。

  1. enum AnimalType: String, Codable {
  2. case cat
  3. case dog
  4. }
  5. struct Animal: Codable {
  6. var name: String
  7. var age: String
  8. }
  9. let pets: [AnimalType: Animal] = [
  10. .cat: .init(name: "Mao", age: "3"),
  11. .dog: .init(name: "Biu", age: "2")
  12. ]
  13. let petsData = try! JSONEncoder().encode(pets)
  14. print(String(decoding: petsData, as: UTF8.self))
  15. // ["cat",{"name":"Mao","age":"3"},"dog",{"name":"Biu","age":"2"}]

打印的结果与预期不符,因为非 String/Int 类型的 key 值,Swift 在转换时无法正确处理。如果我们将这段编码后的字符串以 JSON 形式向服务端传参时,就会出现错误。为此,我们不得不做额外的工作来进行数据转换。

Swift 5.6 新增的 CodingKeyRepresentable 协议很好的解决了这个问题,它支持自定义 key 值的数据类型。我们让 AnimalType 遵循该协议,再次打印的结果与预期一致:

  1. // {"dog":{"name":"Biu","age":"2"},"cat":{"name":"Mao","age":"3"}}

any 关键字

any 和 Any、AnyObject、AnyClass 很像,但它们的关系就像雷锋和雷峰塔。Any 开头的一般表示的是擦除类型信息的类型,any 是一个关键字,用来修饰一种特殊的类型:存在类型(existential types)。

存在类型是一个比较抽象的概念,如果一定要给它下个定义,我会这样描述它:作为类型的协议。比如下面的代码,

  1. protocol UIMode {
  2. var color: Color { get set }
  3. }
  4. struct LightMode: UIMode {
  5. var color: Color = .white
  6. }
  7. struct DarkMode: UIMode {
  8. var color: Color = .black
  9. }
  10. struct ModeManager {
  11. var mode: UIMode
  12. }

在 ModeManager 中 ,协议 UIMode 也被称作存在类型。我们使用一个 DarkMode() 实例来初始化 ModeManager 后,还可以使用 LightMode() 来替换原有的 mode。在编译期,mode 的类型是 UIMode。在运行时,mode 真正的类型是 LightMode、DarkMode 或其它任意遵循该协议的类型,它的值是可以动态分发的。

我们可以把存在类型的值想象成一个盒子,这个盒子可以动态地容纳所有符合该协议类型的值。只要是符合类型的值,相互之间可以动态替换。

我们将上面的代码结合泛型改造一下:

  1. struct ModeManager<T: UIMode> {
  2. var mode: T
  3. }
  4. var manager = ModeManager(mode: DarkMode())
  5. manager.mode = LightMode()

这段代码是无法通过编译的,因为我们在初始化 ModeManager 时传入的是 DarkMode 类型,泛型在编译层面就已经将 mode 约束成了 DarkMode 类型,当我们使用 LightMode 类型的值去改变 mode 时,编译器会报错。

通过对比我们可以看出,当协议作为类型时,它也可以被称为存在类型。如果协议作为泛型的约束,它就无法在运行时动态改变其类型,相应的值只能静态分发。另外,不透明类型中使用 some 关键字修饰的协议,也无法使用不同类型,它要求我们始终使用遵循该协议的特定类型。

动态虽好,但效率不及静态。存在类型带来了性能损耗,但它是我们常用的写法。因此 Swift 5.6 引入了 any 关键字,目的就是提醒我们存在类型的负面影响。同时,我们应该尽可能地避免使用 any。

  1. struct ModeManager {
  2. var mode: any UIMode
  3. }

从 Swift 6 开始,当我们使用存在类型时,编译器会强制要求使用 any 关键字标记,否则会报错。

前文我们提到过,Any 开头的类型一般是擦除类型信息的。它具有一定的动态特性,但会带来一定的性能损耗。这个规律在 SwiftUI 中也是适用的,SwiftUI 中的 AnyView 我们也要慎用。但也不能一概而论,比如 AnyPublisher 我们还是会用到。因此,到底要不要擦除类型信息来换取一定的灵活性,我们要在性能和灵活之间作一个较为平衡的选择。


文章标签:

原文连接:https://juejin.cn/post/7081148979165003807

相关推荐