在 Swift 协议上引用静态变量的正确方法是什么?

What is the proper way to reference a static variable on a Swift Protocol?

假设协议定义如下:

protocol Identifiable {
  static var identifier: String { get }
}
extension Identifiable {
  static var identifier: String { return "Default Id" }
}

引用静态变量的最佳方式是什么?下面的示例说明了访问变量的两种方法。有什么区别,type(of:) 更好吗?

func work<I: Identifiable>(on identifiable: I) {
  let identifier: String = I.identifier
  print("from Protocol: \(identifier)")

  let identiferFromType: String = type(of: identifiable).identifier
  print("using type(of:): \(identiferFromType)")
}

struct Thing: Identifiable {
  static var identifier: String { return "Thing" }
}

work(on: Thing())

在您显示的示例中,没有区别。因为 identifier 是一个协议 requirement,在这两种情况下它都会被动态调度到,因此你不需要担心调用错误的实现。

但是,当 static 计算 属性 中的 self 值时,当 类 符合您的协议时,会出现一个差异。

self in a static method/computed 属性 是调用它的元类型值。因此,当调用 I 时,self 将是 I.self – 这是编译器推断通用占位符 Istatic 类型成为。在 type(of: identifiable) 上调用时,self 将是 identifiable 实例的 动态 元类型值。

为了说明这种差异,请考虑以下示例:

protocol Identifiable {
    static var identifier: String { get }
}

extension Identifiable {
    static var identifier: String { return "\(self)" }
}

func work<I : Identifiable>(on identifiable: I) {
    let identifier = I.identifier
    print("from Protocol: \(identifier)")

    let identiferFromType = type(of: identifiable).identifier
    print("using type(of:): \(identiferFromType)")
}

class C : Identifiable {}
class D : C {}

let d: C = D()

// 'I' inferred to be 'C', 'type(of: d)' is 'D.self'.
work(on: d)

// from Protocol: C
// using type(of:): D

在这种情况下,"which is better" 完全取决于您想要的行为 – 静态或动态。