單例模式

什么是單例模式?

>是開發設計模式(共23種)中的1種

>它可以保證在程序運行過程,一個類只有一個實例(一個對象),而且該實例易于供外界訪問,從而方便地控制了實例個數,并節約系統資源

>使用場合:在整個應用程序中,共享一份資源(這份資源只需要創建初始化1次)

iOS系統中的一些單例

UIApplication(應用程序實例)

NSNotificationCenter(消息中心)

NSFileManager(文件管理)

NSUserDefaults(應用程序設置)

NSURLCache(請求緩存)

NSHTTPCookieStorage(應用程序cookies池)

單例模式的基本實現

1.懶漢模式

// 1.在該類中定義一個靜態的全局變量,防止被外部用extren訪問

staticid_instance;

/**

?static : 修飾變量

?1> 修飾全局變量

?* 全局變量的作用域僅限于當前文件內部,其他文件不能用extren關鍵字訪問

?2> 修飾局部變量 :?

?* 局部變量的生命周期 跟 全局變量 類似

?* 但是不能改變作用域

?* 能保證局部變量永遠只初始化1次,在程序運行過程中,永遠只有1分內存

?*/


/** * 2.重寫它這個類的llocWithZone:方法,確保只為你這個類分配一次內存地址*? alloc方法內部會調用這個方法*/+ (id)allocWithZone:(struct_NSZone *)zone

{

? ? if(_instance == nil) {// 防止頻繁加鎖@synchronized(self) {// 加鎖,避免在多線程中出錯而創建多個對象if(_instance == nil) {// 防止創建多次_instance = [super allocWithZone:zone];

? ? ? ? ? ? }

? ? ? ? }

? ? }

? ? return _instance;

}//3.提供一個shared方法讓外界調用這個單例(一般單例都會提供這個方法),確保只init一次+ (instancetype)sharedMusicTool

{

? ? if(_instance == nil) {// 防止頻繁加鎖? ? ? ? @synchronized(self) {

? ? ? ? ? ? if(_instance == nil) {// 防止創建多次_instance = [[self alloc] init];

? ? ? ? ? ? }

? ? ? ? }

? ? }

? ? return _instance;

}// 4.重寫copyWithZone:方法,避免使用copy時創建多個對象- (id)copyWithZone:(NSZone *)zone

{

? ? return _instance;

}

2.餓漢模式(當類加載到OC運行環境中(內存)時,就會調用一次生成這個單例(一個類只會加載一次))

// 1.在該類中定義一個靜態的全局變量,防止被外部用extren訪問staticid _instance;/**

* 2.重寫它這個類的llocWithZone:方法,這里不用加鎖,因為程序剛啟動,線程還沒加載,不會出現線程不安全的問題

*/+ (id)allocWithZone:(struct_NSZone *)zone

{if(_instance == nil) {// 防止創建多次_instance = [super allocWithZone:zone];

? ? ? ? ? ? }

? ? return _instance;

}// 3.提供一個shared方法讓外界調用這個單例(一般單例都會提供這個方法)+ (instancetype)sharedMusicTool

{

? ? return _instance;

}// 4.重寫copyWithZone:方法,避免使用copy時創建多個對象- (id)copyWithZone:(NSZone *)zone

{

? ? return _instance;

}// 5.重寫load這個類方法,在里面alloc它// 這個方法在程序啟動,加載類的時候會調用一次+ (void)load

{

? _instance = [[self alloc] init];? ? ?

}


用GCD實現單例模式

// 1.staticid _instance;/**

* 2.用GCD的dispatch_once方法重寫

*/+ (id)allocWithZone:(struct_NSZone *)zone

{

? ? static dispatch_once_t onceToken;

? ? dispatch_once(&onceToken, ^{

? ? ? ? _instance = [super allocWithZone:zone];

? ? });

? return _instance;

}// 3.+ (instancetype)sharedMusicTool

{

? ? static dispatch_once_t onceToken;

? ? dispatch_once(&onceToken, ^{

? ? ? ? _instance = [[self alloc] init];

? ? });

? ? return _instance;

}// 4.- (id)copyWithZone:(NSZone *)zone

{

? ? return _instance;

}


非ARC下的單例模式

//前面與ACR一樣,接著重寫以下方法

// 不做release

- (onewayvoid)release {

}// retain之后還是自己一份- (id)retain {

? ? return self;

}// 計數器永遠為1- (NSUInteger)retainCount {

? ? return1;

}// 防止被放進自動計數池釋放- (id)autorelease {

? ? return self;

}


用宏定義實現單例(適配ARC和MRC)

//新建一個.h文件或pch文件,把一下代碼copy進去。在需要實現單例的類中import進去并使用宏(括號里面傳相應的類名)即可

// 在.h文件使用的宏定義#defineHJSingletonH(name) + (instancetype)shared##name;#if__has_feature(objc_arc)// 當前的編譯器環境是ARC// 在.m文件使用的宏定義#defineHJSingletonM(name) \staticid _instace; \

\+ (id)allocWithZone:(struct_NSZone *)zone \

{ \static dispatch_once_t onceToken; \

dispatch_once(&onceToken, ^{ \

_instace = [super allocWithZone:zone]; \

}); \return _instace; \

} \

\+ (instancetype)shared##name \

{ \static dispatch_once_t onceToken; \

dispatch_once(&onceToken, ^{ \

_instace = [[self alloc] init]; \

}); \return _instace; \

} \

\- (id)copyWithZone:(NSZone *)zone \

{ \return _instace; \

}#else// 當前的編譯器環境是MRC// 在.m文件使用的宏定義#defineHJSingletonM(name) \staticid _instace; \

\+ (id)allocWithZone:(struct_NSZone *)zone \

{ \

? ? static dispatch_once_t onceToken; \

? ? dispatch_once(&onceToken, ^{ \

? ? ? ? _instace = [super allocWithZone:zone]; \

? ? }); \

? ? return _instace; \

} \

\+ (instancetype)shared##name \

{ \

? ? static dispatch_once_t onceToken; \

? ? dispatch_once(&onceToken, ^{ \

? ? ? ? _instace = [[self alloc] init]; \

? ? }); \

? ? return _instace; \

} \

\- (id)copyWithZone:(NSZone *)zone \

{ \

? ? return _instace; \

} \- (onewayvoid)release {} \- (id)retain {return self;} \- (NSUInteger)retainCount {return1;} \- (id)autorelease {return self;}#endif

?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,333評論 6 531
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,491評論 3 416
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,263評論 0 374
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,946評論 1 309
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,708評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,186評論 1 324
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,255評論 3 441
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,409評論 0 288
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 48,939評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,774評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,976評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,518評論 5 359
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,209評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,641評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,872評論 1 286
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,650評論 3 391
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,958評論 2 373

推薦閱讀更多精彩內容