線程同步(一)

最近突然想了解了解保持線程同步的方式@synchronized跨释、NSLock僻孝、dispatch_semaphore、NSCondition

@synchronized() :方式一

NSObject *obj = [[NSObject alloc] init];

dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{

@synchronized(obj) {

NSLog(@"需要線程同步的操作1 開始");

NSLog(@"需要線程同步的操作1 結(jié)束");

}

});

dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{

@synchronized(obj) {

NSLog(@"需要線程同步的操作2");

}

});

輸出記

?需要線程同步的操作1 開始

需要線程同步的操作1 結(jié)束

需要線程同步的操作2


方式二:

NSObject *obj = [[NSObject alloc] init];

dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{

@synchronized(obj) {

NSLog(@"需要線程同步的操作1 開始");

NSLog(@"需要線程同步的操作1 結(jié)束");

}

});

dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{

@synchronized(self) {

NSLog(@"需要線程同步的操作2");

}

});

輸出結(jié)果:

需要線程同步的操作2

需要線程同步的操作1 開始

需要線程同步的操作1 結(jié)束

從方式一和二 的輸出不難看出兩者的不同 @synchronized(name) 當(dāng)那么相同時(shí)線程順序輸出,反之,輸入順序不確定



dispatch_semaphore

dispatch_semaphore_t signal = dispatch_semaphore_create(1);

dispatch_time_t overTime = dispatch_time(DISPATCH_TIME_NOW, 3 * NSEC_PER_SEC);

dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{

dispatch_semaphore_wait(signal, overTime);

NSLog(@"需要線程同步的操作1 開始");

NSLog(@"需要線程同步的操作1 結(jié)束");

dispatch_semaphore_signal(signal);

});

dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{

dispatch_semaphore_wait(signal, overTime);

NSLog(@"需要線程同步的操作2");

dispatch_semaphore_signal(signal);

});

需要線程同步的操作1 開始

需要線程同步的操作1 結(jié)束

需要線程同步的操作2


dispatch_semaphore是GCD用來同步的一種方式崇决,與他相關(guān)的共有三個(gè)函數(shù)材诽,分別是dispatch_semaphore_create,dispatch_semaphore_signal恒傻,dispatch_semaphore_wait脸侥。

(1)dispatch_semaphore_create的聲明為:

dispatch_semaphore_t? dispatch_semaphore_create(long value);

傳入的參數(shù)為long,輸出一個(gè)dispatch_semaphore_t類型且值為value的信號量盈厘。

值得注意的是睁枕,這里的傳入的參數(shù)value必須大于或等于0,否則dispatch_semaphore_create會返回NULL扑庞。

(2)dispatch_semaphore_signal的聲明為:

long dispatch_semaphore_signal(dispatch_semaphore_t dsema)

這個(gè)函數(shù)會使傳入的信號量dsema的值加1譬重;

(3) dispatch_semaphore_wait的聲明為:

long dispatch_semaphore_wait(dispatch_semaphore_t dsema, dispatch_time_t timeout);

這個(gè)函數(shù)會使傳入的信號量dsema的值減1罐氨;這個(gè)函數(shù)的作用是這樣的,如果dsema信號量的值大于0滩援,該函數(shù)所處線程就繼續(xù)執(zhí)行下面的語句栅隐,并且將信號量的值減1;如果desema的值為0玩徊,那么這個(gè)函數(shù)就阻塞當(dāng)前線程等待timeout(注意timeout的類型為dispatch_time_t租悄,不能直接傳入整形或float型數(shù)),如果等待的期間desema的值被dispatch_semaphore_signal函數(shù)加1了恩袱,且該函數(shù)(即dispatch_semaphore_wait)所處線程獲得了信號量泣棋,那么就繼續(xù)向下執(zhí)行并將信號量減1。如果等待期間沒有獲取到信號量或者信號量的值一直為0畔塔,那么等到timeout時(shí)潭辈,其所處線程自動執(zhí)行其后語句。

dispatch_semaphore 是信號量澈吨,但當(dāng)信號總量設(shè)為 1 時(shí)也可以當(dāng)作鎖來把敢。在沒有等待情況出現(xiàn)時(shí),它的性能比 pthread_mutex 還要高谅辣,但一旦有等待情況出現(xiàn)時(shí)修赞,性能就會下降許多。相對于 OSSpinLock 來說桑阶,它的優(yōu)勢在于等待時(shí)不會消耗 CPU 資源柏副。

NSLock

NSLock *lock = [[NSLock alloc] init];

dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{

//[lock lock];

[lock lockBeforeDate:[NSDate date]];

NSLog(@"需要線程同步的操作1 開始");

NSLog(@"需要線程同步的操作1 結(jié)束");

[lock unlock];

});

dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{

if ([lock tryLock]) {//嘗試獲取鎖,如果獲取不到返回NO蚣录,不會阻塞該線程

NSLog(@"鎖可用的操作");

[lock unlock];

}else{

NSLog(@"鎖不可用的操作");

}

NSDate *date = [[NSDate alloc] initWithTimeIntervalSinceNow:3];

if ([lock lockBeforeDate:date]) {//嘗試在未來的3s內(nèi)獲取鎖割择,并阻塞該線程,如果3s內(nèi)獲取不到恢復(fù)線程, 返回NO,不會阻塞該線程

NSLog(@"沒有超時(shí)包归,獲得鎖");

[lock unlock];

}else{

NSLog(@"超時(shí)锨推,沒有獲得鎖");

}

});

NSLock是Cocoa提供給我們最基本的鎖對象铅歼,這也是我們經(jīng)常所使用的,除lock和unlock方法外换可,NSLock還提供了tryLock和lockBeforeDate:兩個(gè)方法贮预,前一個(gè)方法會嘗試加鎖,如果鎖不可用(已經(jīng)被鎖住)瓣距,剛并不會阻塞線程响疚,并返回NO。lockBeforeDate:方法會在所指定Date之前嘗試加鎖译荞,如果在指定時(shí)間之前都不能加鎖瓤的,則返回NO。



NSRecursiveLock遞歸鎖


NSRecursiveLock *lock = [[NSRecursiveLock alloc] init];

dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{

static void (^RecursiveMethod)(int);

RecursiveMethod = ^(int value) {

[lock lock];

if (value > 0) {

NSLog(@"value = %d", value);

RecursiveMethod(value - 1);

}

[lock unlock];

};

RecursiveMethod(5);

});

NSRecursiveLock實(shí)際上定義的是一個(gè)遞歸鎖吞歼,這個(gè)鎖可以被同一線程多次請求圈膏,而不會引起死鎖。這主要是用在循環(huán)或遞歸操作中篙骡。


NSConditionLock


NSConditionLock * lock = [NSConditionLock new];

NSMutableArray *products = [NSMutableArray array];

NSInteger HAS_DATA = 1;

NSInteger NO_DATA = 0;

dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{

while (1) {

[lock lockWhenCondition:NO_DATA];

[products addObject:[[NSObject alloc] init]];

NSLog(@"produce a product,總量:%zi",products.count);

[lock unlockWithCondition:HAS_DATA];

}

});

dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{

while (1) {

NSLog(@"wait for product");

[lock lockWhenCondition:HAS_DATA];

[products removeObjectAtIndex:0];

NSLog(@"custome a product");

[lock unlockWithCondition:NO_DATA];

}

});

NSCondition

NSCondition *condition = [[NSCondition alloc] init];

NSMutableArray *products = [NSMutableArray array];

dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{

while (1) {

[condition lock];

if ([products count] == 0) {

NSLog(@"wait for product");

[condition wait];

}

[products removeObjectAtIndex:0];

NSLog(@"custome a product");

[condition unlock];

}

});

dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{

while (1) {

[condition lock];

[products addObject:[[NSObject alloc] init]];

NSLog(@"produce a product,總量:%zi",products.count);

[condition signal];

[condition unlock];

}

});

[condition lock];一般用于多線程同時(shí)訪問稽坤、修改同一個(gè)數(shù)據(jù)源,保證在同一時(shí)間內(nèi)數(shù)據(jù)源只被訪問糯俗、修改一次尿褪,其他線程的命令需要在lock 外等待,只到unlock 得湘,才可訪問

[condition unlock];與lock 同時(shí)使用

[condition wait];讓當(dāng)前線程處于等待狀態(tài)

[condition signal];CPU發(fā)信號告訴線程不用在等待杖玲,可以繼續(xù)執(zhí)行

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市淘正,隨后出現(xiàn)的幾起案子摆马,更是在濱河造成了極大的恐慌,老刑警劉巖跪帝,帶你破解...
    沈念sama閱讀 218,755評論 6 507
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件今膊,死亡現(xiàn)場離奇詭異,居然都是意外死亡伞剑,警方通過查閱死者的電腦和手機(jī)斑唬,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,305評論 3 395
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來黎泣,“玉大人恕刘,你說我怎么就攤上這事∈阋校” “怎么了褐着?”我有些...
    開封第一講書人閱讀 165,138評論 0 355
  • 文/不壞的土叔 我叫張陵,是天一觀的道長托呕。 經(jīng)常有香客問我含蓉,道長频敛,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,791評論 1 295
  • 正文 為了忘掉前任馅扣,我火速辦了婚禮斟赚,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘差油。我一直安慰自己拗军,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,794評論 6 392
  • 文/花漫 我一把揭開白布蓄喇。 她就那樣靜靜地躺著发侵,像睡著了一般。 火紅的嫁衣襯著肌膚如雪妆偏。 梳的紋絲不亂的頭發(fā)上刃鳄,一...
    開封第一講書人閱讀 51,631評論 1 305
  • 那天,我揣著相機(jī)與錄音钱骂,去河邊找鬼铲汪。 笑死,一個(gè)胖子當(dāng)著我的面吹牛罐柳,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播狰住,決...
    沈念sama閱讀 40,362評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼张吉,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了催植?” 一聲冷哼從身側(cè)響起肮蛹,我...
    開封第一講書人閱讀 39,264評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎创南,沒想到半個(gè)月后伦忠,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,724評論 1 315
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡稿辙,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,900評論 3 336
  • 正文 我和宋清朗相戀三年昆码,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片邻储。...
    茶點(diǎn)故事閱讀 40,040評論 1 350
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡赋咽,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出吨娜,到底是詐尸還是另有隱情脓匿,我是刑警寧澤,帶...
    沈念sama閱讀 35,742評論 5 346
  • 正文 年R本政府宣布宦赠,位于F島的核電站陪毡,受9級特大地震影響米母,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜毡琉,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,364評論 3 330
  • 文/蒙蒙 一铁瞒、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧绊起,春花似錦精拟、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,944評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至笋鄙,卻和暖如春师枣,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背萧落。 一陣腳步聲響...
    開封第一講書人閱讀 33,060評論 1 270
  • 我被黑心中介騙來泰國打工践美, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人找岖。 一個(gè)月前我還...
    沈念sama閱讀 48,247評論 3 371
  • 正文 我出身青樓陨倡,卻偏偏與公主長得像,于是被迫代替她去往敵國和親许布。 傳聞我的和親對象是個(gè)殘疾皇子兴革,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,979評論 2 355

推薦閱讀更多精彩內(nèi)容