在线时间:8:00-16:00
迪恩网络APP
随时随地掌握行业动态
扫描二维码
关注迪恩网络微信公众号
最近在开发一个小的应用,遇到了一些Objective-c上面常用的单例模式,但是swift上面还是有一定区别的,反复倒来倒去发现不能按常理(正常的oc to swift的方式)出牌,因此搜索了一些帖子。可能是xcode或者sdk的问题吧(我相信他们不会把未经测试的代码展示,吧?。。。),一些帖子中的代码犯了明显的错误,编译失败。于是有了这篇文章,分享给大家。
原作者实现了一种单例,但是红色代码导致非线程安全: 1 class var sharedInstance:TPScopeManager { 2 get { 3 struct Static { 4 static var instance : TPScopeManager? = nil 5 } 6 7 if !Static.instance { 8 Static.instance = TPScopeManager() 9 } 10 11 return Static.instance! 12 } 13 } 于是有了下面这些解决方案(个人感觉很精彩): 全局常量第一种:直接声明全局变量 let _SingletonSharedInstance = Singleton() class Singleton { ... } 优点:代码最简洁。 缺点:代码开放度较乱
第二种就弥补了上面的缺点 private let _SingletonSharedInstance = Singleton() class Singleton { class var sharedInstance : Singleton { return _SingletonSharedInstance } } 注:因为不支持类型常量(即类的静态常量),所以这里使用了全局常量 这种方式支持延迟(lasy)初始化,因为Swift会延迟初始化全局常量(和变量),并且let关键字是线程安全的。(言外之意:全局变量也是延迟初始化的,但非线程安全?我表示不确定,请大神赐教) Nested struct(估且译为内部struct吧)class Singleton { class var sharedInstance : Singleton { struct Static { static let instance : Singleton = Singleton() } return Static.instance } } 类不支持类型常量(即类的静态常量),但struct支持。利用此,可以达到类似的效果。 原著建议使用内部struct的方式,除非新版本中支持了类型变量 dispatch_once(这个真不能翻译)传统的OC方式在Swift中也是支持的,对比上一种方式,这种方式很明显没有任何优势,但是还是写出来吧 class Singleton { class var sharedInstance : Singleton { struct Static { static var onceToken : dispatch_once_t = 0 static var instance : Singleton? = nil } dispatch_once(&Static.onceToken) { Static.instance = Singleton() } return Static.instance! } } (原理一样,还是用struct支持类型变量这一优势,来把OC的dispatch_once方式平移过来) 如上所述,苹果官方已经明确声明延迟初始化是线程安全的,所以,没有必要再加一层dispatch_once或者类似的保护措施。 全局变量(struct和enum内部的静态成员也同样)的延迟加载本质是dispatch_once,因此如果想使用dispatch_once,不如直接声明一个私有全局变量,即保证了线程安全,也不会使代码过于open 我比较喜欢全局变量方式(当然是经过优化的那种),不用嵌套一层struct,比较简洁。大家喜欢哪种呢?
英语好的可以直接看原帖 http://stackoverflow.com/questions/24024549/dispatch-once-singleton-model-in-swift |
请发表评论