多線程編程里陨闹,會出現(xiàn)共同訪問數(shù)據(jù)造成不安全的問題楞捂,網(wǎng)易的一個面試題很經(jīng)典的詮釋了線程安全。
@property (nonatomic, strong) NSString *target;
?dispatch_queue_t queue = dispatch_queue_create("safe", DISPATCH_QUEUE_CONCURRENT); ??
?for(int i =0; i <10000; i++) {?
? ? ? ?dispatch_async(queue, ^{ ? ? self.target= [NSStringstringWithFormat:@"hello%d",i]; ? ? ? });
?? ?}
上面這樣寫趋厉,會出現(xiàn)EXC_BAD_ACCESS崩潰現(xiàn)象寨闹,意思就是向一個已經(jīng)釋放的對象發(fā)送消息。出現(xiàn)崩潰的原因是過度釋放,可能會出現(xiàn)多個隊列同時釋放.
- (void)setTarget:(NSString*)target {
?? ?[target retain];
? ? [_target release];
? ? _target = target;
}
針對這個問題的解決方案是君账,修飾符strong可以改為weak,nonatomic改為atomic加鎖繁堡,使用串行隊列都行。
不能確定代碼的運行數(shù)據(jù)和結(jié)果乡数,所以才導(dǎo)致崩潰椭蹄,也就是線程不安全的,加鎖是一種很好的解決方案净赴,一般的鎖有NSLock绳矩、@synchronized和dispatch_semaphore等,
@synchronized:(互斥鎖)玖翅,和NSLock相似翼馆,防止不同的線程同時獲取相同的鎖,@synchronized 的作用是創(chuàng)建一個互斥鎖,保證此時沒有其它線程對self對象進行修改金度。這個是objective-c的一個鎖定令牌应媚,防止self對象在同一時間內(nèi)被其它線程訪問,起到線程的保護作用猜极≈薪互斥鎖的優(yōu)點,能有效防止因為多線程搶奪資源造成的數(shù)據(jù)安全問題跟伏,缺點是需要消耗大量的CPU資源丢胚;@synchronized屬于預(yù)防翩瓜,當(dāng)小括號里面的標(biāo)志相同時才滿足互斥,會造成線程阻塞嗜桌,有點事不需要我們創(chuàng)建鎖奥溺;
信號量dispatch_semaphore:終點在于這個函數(shù),dispatch_semaphore_wait首先會先將信號量減一骨宠,并判斷是否大于等于0浮定,如果是,則返回0层亿,并繼續(xù)執(zhí)行后續(xù)代碼桦卒,否則,使線程進入睡眠狀態(tài)匿又,讓出cpu時間方灾。直到信號量大于0或者超時,則線程會被重新喚醒執(zhí)行后續(xù)操作碌更。這個函數(shù)會使傳入的信號量dsema的值減1裕偿;這個函數(shù)的作用是這樣的,如果dsema信號量的值大于0痛单,該函數(shù)所處線程就繼續(xù)執(zhí)行下面的語句嘿棘,并且將信號量的值減1;如果desema的值為0旭绒,那么這個函數(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時房午,其所處線程自動執(zhí)行其后語句;
自旋鎖:死循環(huán)丹允,當(dāng)A線程得到所以后歪沃,B線程想要獲取鎖就要等待A線程釋放鎖,期間一直處于等待狀態(tài)嫌松;