Objective-C设计模式——单例Singleton(对象创建)
單例
和其它語言的單例產不多,可以說是最簡單的一種設計模式了。但是有幾個點需要注意下,單例就是一個類只有一個實例。
所以我們要想辦法阻止該類產生別的實例,一般語言中都會將構造函數寫為private。但是OC中的函數并沒有限定符,所以我們需要用一些小技巧來屏蔽這一點。
?
應用場景
類只能有一個實例,而且必須從一個為人熟知的訪問點對其進行訪問,比如工廠方法。
這個唯一的實例只能通過類的子類化進行擴展,而且擴展的對象不會破壞客戶端代碼。
?
注意
1.OC中單例的實例變量要定義在.m文件
2.OC中單例需要重載allocWithZone:和copyWithZone:方法來防止創建別的實例。
3.單例創建要注意線程安全,不然就可能出現多個實例。
注意問題將會在Demo中講解
?
Demo
首先先來看一個最常規,的不嚴謹的單例實現:
@implementation Singletonstatic Singleton *sharedInstance;-(Singleton *)sharedInstance {if(sharedInstance){sharedInstance = [Singleton new];}return sharedInstance; }@end這看似好像是可以得到單例對象了,但是這可以說是單例的一種變形。絕不能說這就是單例,因為我們可以輕松地通過其他方式來創建對象。
所以我們還要我修改allocWithZone:和copyWithZone:方法(alloc 和 copy 方法實際上就是調用這兩個方法)
-(id)copyWithZone:(NSZone *)zone {return [[self class] sharedInstance]; }+(id)allocWithZone:(struct _NSZone *)zone {return [self sharedInstance]; }可是這就出現另一個問題,在sharedInstance方法里面我們實際調用過allocWithZone:(new 調用 alloc),但是它的alloc被我們重寫了,這就會出現錯誤。所以我們需要修改sharedInstance方法
+(Singleton *)sharedInstance {if(sharedInstance){sharedInstance = [[super allocWithZone:NULL] init];}return sharedInstance; }這樣就可以順利的返回單例了,而且無法通過其它方式產生實例對象。
看似完美了實際還會有問題出現,因為現在是非線程安全的,可能存在同一時間創建多個實例的情況,所以修改如下
+(instancetype)sharedInstance {static dispatch_once_t once;dispatch_once(&once, ^{sharedInstance = [[super allocWithZone:NULL] init];});return sharedInstance; }客戶端代碼如下:
Singleton *singleton = [Singleton sharedInstance];Singleton *singleton2 = [[Singleton alloc] init];Singleton *singleton3 = [singleton copy];[singleton print];[singleton2 print];[singleton3 print];輸出結果:
2015-07-21 21:10:32.797 Singleton[42537:10347987] 0x7fff5fbff7a8 2015-07-21 21:10:32.798 Singleton[42537:10347987] 0x7fff5fbff7a8 2015-07-21 21:10:32.798 Singleton[42537:10347987] 0x7fff5fbff7a8可以看到內存都指向了同一地址。
轉載于:https://www.cnblogs.com/pb89/p/4708785.html
總結
以上是生活随笔為你收集整理的Objective-C设计模式——单例Singleton(对象创建)的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 梦到车打不着火怎么回事
- 下一篇: ASP.NET 5 Beta7发布