因此我决定将各种单例都写一变,然后在运行时加入断点来观测。
  分析了每个stack trace的记录后,我发现了有趣的东西——证据!
  来看看截图:
  使用全局单例方法


  
  使用单行单例方法

  第一张图片展示了使用全局实例时的stack trace。标红的地方需要注意。在调用Kraken单例之前,先调用了swift_once,接下来是swift_once_block_invoke。Apple之前在文档中已经说过,“懒实例化”的全局变量会被自动放在dispatch_once块中,我们可以假定说的是这个东西。
  了解了这些知识,我们来看看漂亮的单行单例方法。如图所示,调用完全一样。这样,我们有了证据证明单行单例方法是正确的。
  不要忘记设置初始化方法为私有
  @davedelong,Apple的Framework传道者,善意地提醒我:必须保证init方法的私有性,只有这样,才能保证单例是真正的,避免外部对象通过访问init方法创建单例类的其他实例。由于Swift中的所有对象都是由公共的初始化方法创建的,我们需要重写自己的init方法,并设置其为私有的。这很简单,而且不会破坏到我们优雅的单行单例方法。
  class TheOneAndOnlyKraken {
  static let sharedInstance = TheOneAndOnlyKraken()
  private init() {} //This prevents others from using the default '()' initializer for this class.
  }
  这样做可以保证编译器在某个类尝试使用()来初始化TheOneAndOnlyKraken时,抛出错误:

  是这样,我们的单行单例,非常完美!
  结论
  这里回复一下jtbandes在“top rated answer to swift singletons on Stack Overflow”这个帖子中的问题:我也找不到哪里有文档证明let语句可以带来线程安全性的好处。我记得在去年参加WWDC的时候有类似的说法,没办法保证读者或各位Googler也偶遇到这个说法。希望这个帖子能帮助大家理解为什么单行单例在Swift中是正确的方法。