單例

單例的定義

單例模式確保某一個(gè)類只有一個(gè)實(shí)例,而且自行實(shí)例化并向整個(gè)系統(tǒng)提供這個(gè)實(shí)例單例模式琐脏。

注:單例模式只應(yīng)在有真正的“單一實(shí)例”的需求時(shí)才可使用攒砖。

單例的創(chuàng)建

單線程

+ (instancetype)shareInstance{
    static Singleton* single;
    if(!single) {      
        single = [[self alloc] init];   
    }
    return single;
}

注:上面是單線程的單例。如果有多線程同時(shí)訪問日裙,不能保證為同一個(gè)實(shí)例吹艇。

多線程

使用@synchronized加鎖來創(chuàng)建單例

+ (instancetype)shareInstance{
    static Singleton* single;
    @synchronized(self) {
        if(!single) {           
            single = [[self alloc] init];       
        }    
    }
    return single;
}

使用@synchronized雖然解決了多線程的問題,但有性能問題昂拂。因?yàn)榧渔i只有在single未創(chuàng)建時(shí)才有必要受神。如果single已經(jīng)創(chuàng)建,就不需要加鎖格侯,但程序會(huì)再次進(jìn)行synchronized驗(yàn)證鼻听,浪費(fèi)性能。
dispatch_once 能做到既解決同步多線程問題而又不影響性能联四。

+ (SingletonManager*)shareInstance {
    static Singleton* single;
    static dispatch_once_t token;
    dispatch_once(&token, ^{
        if(single == nil) {
            single = [[self alloc] init];
        }
    });
    return single;
}
dispatch_once的原理

dispatch_once主要是根據(jù)onceToken的值來決定怎么去執(zhí)行代碼撑碴。

  • 當(dāng)onceToken= 0時(shí),線程執(zhí)行dispatch_once的block中代碼
  • 當(dāng)onceToken= -1時(shí)朝墩,線程跳過dispatch_once的block中代碼不執(zhí)行
  • 當(dāng)onceToken為其他值時(shí)醉拓,線程被線程被阻塞,等待onceToken值改變

#ifdef __BLOCKS__
__OSX_AVAILABLE_STARTING(__MAC_10_6,__IPHONE_4_0)
DISPATCH_EXPORT DISPATCH_NONNULL_ALL DISPATCH_NOTHROW
void
dispatch_once(dispatch_once_t *predicate, dispatch_block_t block);
 
DISPATCH_INLINE DISPATCH_ALWAYS_INLINE DISPATCH_NONNULL_ALL DISPATCH_NOTHROW
void
_dispatch_once(dispatch_once_t *predicate, dispatch_block_t block)
{
    if (DISPATCH_EXPECT(*predicate, ~0l) != ~0l) {
        dispatch_once(predicate, block);
    }
}
#undef dispatch_once
#define dispatch_once _dispatch_once

dispatch_once_t 發(fā)現(xiàn)其是 typedef long類型

靜態(tài)變量在程序運(yùn)行期間只被初始化一次,然后其在下一次被訪問時(shí)亿卤,其值都是上次的值愤兵,其在除了這個(gè)初始化方法以外的任何地方都不能直接修改這兩個(gè)變量的值。這是單例只被初始化一次的前提排吴。

先聲明了dispatch_once 函數(shù)秆乳,下面又實(shí)現(xiàn)了_dispatch_once 函數(shù)。
先取消dispatch_once 的定義傍念, 然后把_dispatch_once 定義為dispatch_once矫夷。
所以用戶調(diào)用 dispatch_once 函數(shù),實(shí)際上調(diào)用的是_dispatch_once 函數(shù)憋槐;
而真正的dispatch_once 函數(shù)是在_dispatch_once 內(nèi)調(diào)用的双藕。
有一個(gè)判斷條件是一個(gè)宏DISPATCH_EXPECT,而判斷條件為 DISPATCH_EXPECT(*predicate, ~0l) 阳仔,就是說忧陪,*predicate 很可能是 ~0l ,而當(dāng) DISPATCH_EXPECT(*predicate, ~0l) 不是 ~0! 時(shí)才調(diào)用真正的dispatch_once 函數(shù)近范。
~0l的意思是長整型0按位取反嘶摊,其實(shí)就是長整型的-1

DISPATCH_EXPECT的宏定義為

#if __GNUC__
#define DISPATCH_EXPECT(x, v) __builtin_expect((x), (v))
#else
#define DISPATCH_EXPECT(x, v) (x)

意思是如果沒有定義__GNUC__的話 DISPATCH_EXPECT(x, v)就是第一個(gè)參數(shù)(x)
__GNUC__只代表gcc的主版本號(hào)
__builtin_expect 作用是允許程序員將最有可能執(zhí)行的分支告訴編譯器评矩。
這個(gè)指令的寫法為:__builtin_expect(EXP, N), 意思是:EXP==N的概率很大叶堆。
目的是將“分支轉(zhuǎn)移”的信息提供給編譯器,這樣編譯器可以對(duì)代碼進(jìn)行優(yōu)化斥杜,以減少指令跳轉(zhuǎn)帶來的性能下降虱颗。

總結(jié)

第一次運(yùn)行,predicate的值是默認(rèn)值0蔗喂,線程執(zhí)行dispatch_once忘渔,執(zhí)行完之后predicate=-1,當(dāng)另一個(gè)線程再次訪問缰儿,predicate=-1畦粮,不執(zhí)行dispatch_once
如果有兩個(gè)進(jìn)程同時(shí)運(yùn)行到dispatch_once 方法時(shí)乖阵,這個(gè)兩個(gè)進(jìn)程獲取到的predicate 值都是0宣赔,那么最終兩個(gè)進(jìn)程都會(huì)調(diào)用 最原始那個(gè)dispatch_once 函數(shù)。
猜測(cè):在極端情況下瞪浸,也有可能出現(xiàn)2個(gè)實(shí)例拉背。dispatch_once也不是線程安全。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末默终,一起剝皮案震驚了整個(gè)濱河市椅棺,隨后出現(xiàn)的幾起案子犁罩,更是在濱河造成了極大的恐慌,老刑警劉巖两疚,帶你破解...
    沈念sama閱讀 217,734評(píng)論 6 505
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件床估,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡诱渤,警方通過查閱死者的電腦和手機(jī)丐巫,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,931評(píng)論 3 394
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來勺美,“玉大人递胧,你說我怎么就攤上這事∩娜祝” “怎么了缎脾?”我有些...
    開封第一講書人閱讀 164,133評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵,是天一觀的道長占卧。 經(jīng)常有香客問我遗菠,道長,這世上最難降的妖魔是什么华蜒? 我笑而不...
    開封第一講書人閱讀 58,532評(píng)論 1 293
  • 正文 為了忘掉前任辙纬,我火速辦了婚禮,結(jié)果婚禮上叭喜,老公的妹妹穿的比我還像新娘贺拣。我一直安慰自己,他們只是感情好捂蕴,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,585評(píng)論 6 392
  • 文/花漫 我一把揭開白布譬涡。 她就那樣靜靜地躺著,像睡著了一般启绰。 火紅的嫁衣襯著肌膚如雪昂儒。 梳的紋絲不亂的頭發(fā)上沟使,一...
    開封第一講書人閱讀 51,462評(píng)論 1 302
  • 那天委可,我揣著相機(jī)與錄音,去河邊找鬼腊嗡。 笑死着倾,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的燕少。 我是一名探鬼主播卡者,決...
    沈念sama閱讀 40,262評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼客们!你這毒婦竟也來了崇决?” 一聲冷哼從身側(cè)響起材诽,我...
    開封第一講書人閱讀 39,153評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎恒傻,沒想到半個(gè)月后脸侥,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,587評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡盈厘,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,792評(píng)論 3 336
  • 正文 我和宋清朗相戀三年睁枕,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片沸手。...
    茶點(diǎn)故事閱讀 39,919評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡外遇,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出契吉,到底是詐尸還是另有隱情跳仿,我是刑警寧澤,帶...
    沈念sama閱讀 35,635評(píng)論 5 345
  • 正文 年R本政府宣布栅隐,位于F島的核電站塔嬉,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏租悄。R本人自食惡果不足惜谨究,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,237評(píng)論 3 329
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望泣棋。 院中可真熱鬧胶哲,春花似錦、人聲如沸潭辈。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,855評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽把敢。三九已至寄摆,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間修赞,已是汗流浹背婶恼。 一陣腳步聲響...
    開封第一講書人閱讀 32,983評(píng)論 1 269
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留柏副,地道東北人勾邦。 一個(gè)月前我還...
    沈念sama閱讀 48,048評(píng)論 3 370
  • 正文 我出身青樓,卻偏偏與公主長得像割择,于是被迫代替她去往敵國和親眷篇。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,864評(píng)論 2 354