熱線電話:0755-23712116
郵箱:contact@shuangyi-tech.com
地址:深圳市寶安區(qū)沙井街道后亭茅洲山工業(yè)園工業(yè)大廈全至科技創(chuàng)新園科創(chuàng)大廈2層2A
信號量用來干嘛的呢?搜尋答案的話,很多人都會告訴你主要用于線程同步的,意思就是線程通信的。簡單來說,比如我運行了2個線程A和B,但是我希望B線程在A線程之前執(zhí)行,那么我們就可以用信號量來處理。有些人可能會疑惑,那么麻煩干嘛?你不是要B線程先執(zhí)行嗎?那么我讓A線程休眠一點時間不就可以了嗎?沒錯,這個思路是可以的,但是如果B線程也因為某些原因(比如硬件,操作系統(tǒng)的原因)導致延緩執(zhí)行了,這該怎么辦?到底A線程該休眠多少時間合適呢?所以正確的做法就是在B線程阻塞,A線程去喚醒這個阻塞線程。
看到這兒,看過我前面文章的朋友可能一眼就看出來了這個不就是前面講的生產(chǎn)消費者模型提到的用法嗎?
沒錯,信號量的實現(xiàn)也是靠條件變量和互斥鎖。
所以雖然C++中并沒有在語言級別上支持信號量,但同樣的我們可以利用以上兩個來自己實現(xiàn)一個。
這里我也不得不提一句,條件變量和互斥鎖組合使用真的非常強大,生產(chǎn)消費者模型中用到了,線程池中用到了,現(xiàn)在說的信號量也用到了,所以大家一定要好好掌握條件變量和互斥鎖的使用,它們倆是你在多線程世界中縱橫捭闔的利劍。
那么我們如何用C++來實現(xiàn)一個信號量呢?
#ifndef _SEMAPHORE_H
#define _SEMAPHORE_H
#include <mutex>
#include <condition_variable>
using namespace std;
class Semaphore
{
public:
Semaphore(long count = 0) : count(count) {}
//V操作,喚醒
void signal()
{
unique_lock<mutex> unique(mt);
++count;
if (count <= 0)
cond.notify_one();
}
//P操作,阻塞
void wait()
{
unique_lock<mutex> unique(mt);
--count;
if (count < 0)
cond.wait(unique);
}
private:
mutex mt;
condition_variable cond;
long count;
};
#endif
信號量里面用到了一個叫PV操作的東西,P操作時阻塞,一般用wait()函數(shù),V操作是喚醒,一般用singal()函數(shù),至于不叫WS操作,反而為什么叫PV操作呢?網(wǎng)上說是因為提出這一系統(tǒng)方法的人狄克斯特拉用荷蘭文定義的,因為在荷蘭文中,通過叫passeren,釋放叫vrijgeven,PV操作因此得名。對我們來說,這些也沒有太大的意義,記住這些定義就好了,畢竟定義這種東西,是不以我們的意志為轉移的。
寫好了信號量的接口,那我們如何使用這個信號量呢?這個就需要我們在外部寫一個多線程的調用函數(shù)來調用。
#include "semaphore.h"
#include <thread>
#include <iostream>
using namespace std;
Semaphore sem(0);
void funA()
{
sem.wait();
//do something
cout << "funA" << endl;
}
void funB()
{
this_thread::sleep_for(chrono::seconds(1));
//do something
cout << "funB" << endl;
sem.signal();
}
int main()
{
thread t1(funA);
thread t2(funB);
t1.join();
t2.join();
}
這里我們想讓funB線程運行,然后再運行funA,多線程是通過時間片輪詢來執(zhí)行的。
假設先開始跑funA,執(zhí)行到sem.wait()的時候,進入wait函數(shù)可知,count減1,小于0,會發(fā)生阻塞,等待其他線程喚醒。
然后就會切換到funB,這里即使休眠了1秒也不會切換到funA,因為那邊阻塞了,沒有其他線程喚醒的話就會一直阻塞。funB休眠完之后,就會打印出結果,然后執(zhí)行sem.signal(),進入signal函數(shù)可知,count加1,小于等于0,會喚醒其他阻塞的線程。
然后再切到funA,執(zhí)行后面的操作,打印出結果。
所以這個就一定能保證funB先執(zhí)行,funA后執(zhí)行。當然前提是初始化信號量對象的時候,要初始化為0。
Semaphore sem(0);
信號量用在多線程多任務同步的,一個線程完成了某一個動作就通過信號量告訴別的線程,別的線程再進行某些動作。像這里funB完成任務之后就通過信號量的PV操作告訴funA線程可以開始任務了。
最后需要注意的是,信號量不僅可以用于進程也可用于線程,它比條件變量要復雜很多,條件變量僅限于線程內使用,至于進程間如何使用信號量通信,后期我們在討論。