異步變同步,信號量的簡單應(yīng)用

最近做一些設(shè)備指紋的收集工作缕棵,使用到信號量孵班,是為了解決異步API同步使用的問題。先來看需求:

陀螺儀加速計的數(shù)據(jù)收集招驴,主要有兩種方法篙程,一種是pull方法,如下:

- (CMRotationRate)getGyroData
{
    CMRotationRate data;
    data.x = 0.f;
    data.y = 0.f;
    data.z = 0.f;
    if ([self isGyroAvailable] && ![self isGyroActive]){//判斷陀螺儀是否可用别厘,是否活躍
        self.gyroUpdateInterval = 0.01;
        [self startGyroUpdates];//開始更新陀螺儀數(shù)據(jù)
    }
    if ([self isGyroAvailable]){
        data = self.gyroData.rotationRate;//獲取到陀螺儀數(shù)據(jù)
        NSLog(@"X = %.04f",data.x);
        NSLog(@"Y = %.04f",data.y);
        NSLog(@"Z = %.04f",data.z);
        return data;
    }else{
        NSAssert(!(data.x || data.y || data.z), @"陀螺儀未啟動或者不可用虱饿,數(shù)據(jù)獲取失敗");
        return data;
    }
}

但是發(fā)現(xiàn),第一次獲取的數(shù)據(jù)總是0触趴。需要多次調(diào)用(一段時間間隔氮发,很小很小)之后才可以獲取到數(shù)據(jù)冗懦,這樣就造成了一個問題爽冕,我只想要調(diào)用方法就返回陀螺儀數(shù)據(jù),但是你給我返回0.0000 WTF??

那么披蕉,再來看陀螺儀的push方法:

 - (void)startGyroUpdatesToQueue:(NSOperationQueue *)queue withHandler:(CMGyroHandler)handler API_UNAVAILABLE(tvos);

Discussion 是這樣寫的

Starts gyro updates, providing data to
the given handler through the given queue.


開始更新陀螺儀數(shù)據(jù),通過block在陀螺儀數(shù)據(jù)更新后回調(diào)返回數(shù)據(jù)扇售。

如此前塔,真正的需求出來了,如何在block回調(diào)之后承冰,return數(shù)據(jù)呢?歸類起來這是一個異步block變同步方法返回的需求食零,終于困乒,主角信號量要出現(xiàn)了!

先看GCD中的信號量API贰谣,非常簡單娜搂,只有三個方法

[圖片上傳失敗...(image-a87808-1538273800178)]

主要方法:

  • 1.dispatch_semaphore_create //創(chuàng)建一個信號量semaphore
  • 2.dispatch_semaphore_wait //信號量等待
  • 3.dispatch_semaphore_signal //發(fā)送一個信號

信號量使用也很簡單:

  • 信號量在創(chuàng)建的時候需要傳入一個long型value,內(nèi)部會保存這個value作為初始value吱抚,value 做加減變化后百宇,還會保存一份當(dāng)前的 value。
  • 信號的 wait 和 signal 是互逆的兩個操作秘豹。signal 會將 value 加一携御。如果 value 大于 0,wait將 value 減一既绕;此時如果 value 小于零就一直等待啄刹。
  • 初始 value 必須大于等于 0,如果為 0 并隨后調(diào)用 wait 方法凄贩,線程將被阻塞直到別的線程調(diào)用了 signal 方法誓军。

簡單了解了信號量的使用,異步block 同步返回的需求就很好解決了疲扎,結(jié)合信號量昵时,實現(xiàn)代碼如下

- (CMRotationRate)syncGetGyroData
{
    __block CMRotationRate data;//保存block數(shù)據(jù)
    data.x = 0.f;
    data.y = 0.f;
    data.z = 0.f;
    dispatch_semaphore_t sema = dispatch_semaphore_create(0);//創(chuàng)建信號量傳入 0,此時遇如果wait會阻塞線程
    [self startGyroUpdatesToQueue:[[NSOperationQueue alloc] init] withHandler:^(CMGyroData * _Nullable gyroData, NSError * _Nullable error) {
        data = gyroData.rotationRate;
        NSLog(@"X = %.04f",gyroData.rotationRate.x);
        NSLog(@"Y = %.04f",gyroData.rotationRate.y);
        NSLog(@"Z = %.04f",gyroData.rotationRate.z);
        dispatch_semaphore_signal(sema);// 信號量 value 加一
    }];

    dispatch_semaphore_wait(sema, DISPATCH_TIME_FOREVER);//信號量為0時會阻塞線程椒丧,等待signle
    return data;
}

以上就完美的解決了異步block同步返回的問題壹甥。信號量使用起來很簡單。
但是需要注意
block回調(diào)的線程瓜挽,如果和syncGetGyroData方法的執(zhí)行線程相同的話盹廷,會造成線程死鎖,這里block回調(diào)不在主線程,所以可以這樣處理久橙,也算是一種小技巧.
另外同樣的需求俄占,也可以借助GCD dispatch_group實現(xiàn):

- (CMRotationRate)syncGetGyroDataByGroup
{
    __block CMRotationRate data;//保存block數(shù)據(jù)
    data.x = 0.f;
    data.y = 0.f;
    data.z = 0.f;
    dispatch_group_t group = dispatch_group_create();
    dispatch_group_enter(group);
    [self startGyroUpdatesToQueue:[[NSOperationQueue alloc] init] withHandler:^(CMGyroData * _Nullable gyroData, NSError * _Nullable error) {
        data = gyroData.rotationRate;
        NSLog(@"X = %.04f",gyroData.rotationRate.x);
        NSLog(@"Y = %.04f",gyroData.rotationRate.y);
        NSLog(@"Z = %.04f",gyroData.rotationRate.z);
        dispatch_group_leave(group);
    }];
    dispatch_group_wait(group, DISPATCH_TIME_FOREVER);
    return data;
}

信號量還可以用于控制線程并發(fā)數(shù)量,這里就不做詳細(xì)介紹了淆衷,可以參考
https://www.cnblogs.com/yajunLi/p/6274282.html

總結(jié):

這次需求缸榄,使用到了信號量。知其然不知其所以然祝拯,對于信號量的實現(xiàn)甚带,還不是很清楚她肯。工作中接觸新東西,很多情況都是這個過程鹰贵。所以下一步打算仔細(xì)了解一下信號量晴氨,GCD源碼,我來了碉输!

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末籽前,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子敷钾,更是在濱河造成了極大的恐慌枝哄,老刑警劉巖,帶你破解...
    沈念sama閱讀 206,378評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件阻荒,死亡現(xiàn)場離奇詭異挠锥,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)侨赡,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,356評論 2 382
  • 文/潘曉璐 我一進(jìn)店門蓖租,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人辆毡,你說我怎么就攤上這事菜秦。” “怎么了舶掖?”我有些...
    開封第一講書人閱讀 152,702評論 0 342
  • 文/不壞的土叔 我叫張陵球昨,是天一觀的道長。 經(jīng)常有香客問我眨攘,道長主慰,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 55,259評論 1 279
  • 正文 為了忘掉前任鲫售,我火速辦了婚禮共螺,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘情竹。我一直安慰自己藐不,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 64,263評論 5 371
  • 文/花漫 我一把揭開白布秦效。 她就那樣靜靜地躺著雏蛮,像睡著了一般。 火紅的嫁衣襯著肌膚如雪阱州。 梳的紋絲不亂的頭發(fā)上挑秉,一...
    開封第一講書人閱讀 49,036評論 1 285
  • 那天,我揣著相機(jī)與錄音苔货,去河邊找鬼犀概。 笑死立哑,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的姻灶。 我是一名探鬼主播铛绰,決...
    沈念sama閱讀 38,349評論 3 400
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼木蹬!你這毒婦竟也來了至耻?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 36,979評論 0 259
  • 序言:老撾萬榮一對情侶失蹤镊叁,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后走触,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體晦譬,經(jīng)...
    沈念sama閱讀 43,469評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 35,938評論 2 323
  • 正文 我和宋清朗相戀三年互广,在試婚紗的時候發(fā)現(xiàn)自己被綠了敛腌。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 38,059評論 1 333
  • 序言:一個原本活蹦亂跳的男人離奇死亡惫皱,死狀恐怖像樊,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情旅敷,我是刑警寧澤生棍,帶...
    沈念sama閱讀 33,703評論 4 323
  • 正文 年R本政府宣布,位于F島的核電站媳谁,受9級特大地震影響涂滴,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜晴音,卻給世界環(huán)境...
    茶點故事閱讀 39,257評論 3 307
  • 文/蒙蒙 一柔纵、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧锤躁,春花似錦搁料、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,262評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至觉啊,卻和暖如春拣宏,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背杠人。 一陣腳步聲響...
    開封第一講書人閱讀 31,485評論 1 262
  • 我被黑心中介騙來泰國打工勋乾, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留宋下,地道東北人。 一個月前我還...
    沈念sama閱讀 45,501評論 2 354
  • 正文 我出身青樓辑莫,卻偏偏與公主長得像学歧,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子各吨,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 42,792評論 2 345

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

  • 放學(xué)回家第一天枝笨,有個叫做虛榮的小種子在心中悄然發(fā)芽。原因很簡單只是因為是一個大學(xué)生揭蜒,要知道在農(nóng)村大學(xué)生并不常見横浑,...
    生活和小珂閱讀 234評論 1 0
  • 鐘年人閱讀 251評論 2 8
  • 朝陽還未升起 天色還很熹微 紛紛擾擾的交通 朦朦朧朧的乘客 當(dāng)躺在戀人的懷里熟睡 當(dāng)拿著大包小包前行 我們?yōu)榱松?..
    日晟昌閱讀 278評論 0 0
  • three.js版本不同,效果會有些差異屉更,我的這個版本中粒子貼圖有問題徙融,邊緣沒有透明,會互相遮擋瑰谜。也許可以通過修改...
    狂暴機(jī)甲閱讀 496評論 0 0