成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

深究 Linux 多線程中的信號量 Semaphore

系統 Linux
Semaphore,對多線程有過了解的人都聽說過,一般我們解釋為“信號量”??墒?,這個單詞對我們來說還是比較陌生,它和另一個單詞 Singal(信號)什么關系呢?想要真正理解這個概念,必須得從它的翻譯開始。

理解 Semaphore,從一個好的翻譯開始

Semaphore,對多線程有過了解的人都聽說過,一般我們解釋為“信號量”??墒牵@個單詞對我們來說還是比較陌生,它和另一個單詞 Singal(信號)什么關系呢?想要真正理解這個概念,必須得從它的翻譯開始。事實上,Semaphore 最好的翻譯應該為“信號計數量”,承認了這一點,想必你也清楚了:它和 Signal 不是一回事!

 

深究 Linux 多線程中的信號量 Semaphore
劍橋詞典翻譯,并不容易理解

信號:簡單來說就是消息,是由用戶、系統或者進程發送給目標進程的信息,用來通知目標進程某個狀態的改變或系統異常,對應的是異步的場景(我之前的文章有詳細介紹過)。

信號量:首先是一個變量,其次是計數器。它是多線程環境下使用的一種設施,信號量在創建時需要設置一個初始值,表示同時可以有幾個任務(線程)可以訪問某一塊共享資源。

  • 一個任務要想訪問共享資源,前提是信號量大于0,當該任務成功獲得資源后,將信號量的值減 1;
  • 若當前信號量的值小于 0,表明無法獲得信號量,該任務必須被掛起,等待信號量恢復為正值的那一刻;
  • 當任務執行完之后,必須釋放信號量,對應操作就是信號量的值加 1。

另外,對信號量的操作(加、減)都是原子的?;コ怄i(Mutex)就是信號量初始值為 1 時的特殊情形,即同時只能有一個任務可以訪問共享資源區。

 

深究 Linux 多線程中的信號量 Semaphore

Semaphore 再理解

我們來設想這樣一個場景(上圖):假如北京的國家大劇院有一場免費的音樂會演出,可是現在正值疫情期間,劇院規定:劇院觀眾總人數要限制,但是允許大家中途退場,把票給其他人,其他人可以中途進場。于是,第一批先到的人從劇院門口票箱中取到了票,然后進場欣賞演出。后到的人就因為劇院滿了,在門口等待。過了一段時間,有人嫌節目太無聊了,提前退場了,退場時他把門票放回去了。這樣,其他人拿著這個人的票進場了。隨后,又有人退場了,但是他忘記把票放回去了。這也沒關系,大不了劇院內可容納的總人數少了一個罷了。

上面的例子中,音樂會現場就是一塊共享資源區,觀眾就是任務(線程),而票箱中的門票數就是信號量。信號量用作并發量限制,由于總的門票數是固定的,所以不會出現音樂廳被擠爆的情況。

上述的例子中,我們允許退場的觀眾把票帶走,這是為什么呢?因為劇院工作人員可以隨時在票箱里補充些門票呀(線程生產者)。說到這,你們是不是有點似曾相識呀?對啰,就是線程池,但還是有些不同,你們自己品味吧。

Semaphore 實操練習

信號量類型為 sem_t,類型及相關操作定義在頭文件 semaphore.h 中,

創建信號量

  1. int sem_init(sem_t *sem, int pshared, unsigned int value); 

信號量的值加 1

  1. int sem_post(sem_t *sem); 

信號量的值減 1

  1. int sem_wait(sem_t *sem); 

信號量銷毀

  1. int sem_destroy(sem_t *sem); 

具體參數含義及返回值,這里就不贅述了。下面展示了一個例子:

你總共有三種類型的下載任務(類型 id 為 1、2、3),每次從鍵盤讀取一種類型的任務進行下載,但是 CPU 最多可以同時執行 2 個下載任務(創建兩個線程)。

 

  1. #include <stdio.h> 
  2. #include <pthread.h> 
  3. #include <semaphore.h> 
  4. #define MAXNUM (2) 
  5. sem_t semDownload; 
  6. pthread_t a_thread, b_thread, c_thread; 
  7. int g_phreadNum = 1; 
  8.  
  9. void func1(void *arg) 
  10.     // 等待信號量的值 > 0 
  11.     sem_wait(&semDownload); 
  12.     printf("============== Downloading taskType 1 ============== \n"); 
  13.     sleep(5); 
  14.     printf("============== Finished taskType 1 ============== \n"); 
  15.     g_phreadNum--; 
  16.     // 等待線程結束 
  17.     pthread_join(a_thread, NULL); 
  18.  
  19. void func2(void *arg) 
  20.     sem_wait(&semDownload); 
  21.     printf("============== Downloading taskType 2 ============== \n"); 
  22.     sleep(3); 
  23.     printf("============== Finished taskType 2 ============== \n"); 
  24.     g_phreadNum--; 
  25.     pthread_join(b_thread, NULL); 
  26.  
  27. void func3(void *arg) 
  28.     sem_wait(&semDownload); 
  29.     printf("============== Downloading taskType 3 ============== \n"); 
  30.     sleep(1); 
  31.     printf("============== Finished taskType 3 ============== \n"); 
  32.     g_phreadNum--; 
  33.     pthread_join(c_thread, NULL); 
  34.  
  35. int main() 
  36.     // 初始化信號量 
  37.     sem_init(&semDownload, 0, 0); 
  38.     int taskTypeId; 
  39.     while (scanf("%d", &taskTypeId) != EOF) 
  40.     { 
  41.         // 輸入 0, 測試程序是否能正常退出 
  42.         if (taskTypeId == 0 && g_phreadNum <= 1) 
  43.         { 
  44.             break; 
  45.         } else if (taskTypeId == 0) 
  46.         { 
  47.             printf("Can not quit, current running thread num is %d\n", g_phreadNum - 1); 
  48.         } 
  49.         printf("your choose Downloading taskType %d\n", taskTypeId); 
  50.         // 線程數超過 2 個則不下載 
  51.         if (g_phreadNum > MAXNUM) 
  52.         { 
  53.             printf("!!! You've reached the max number of threads !!!\n"); 
  54.             continue
  55.         } 
  56.         // 用戶選擇下載 Task 
  57.         switch (taskTypeId) 
  58.         { 
  59.         case 1: 
  60.             // 創建線程 1 
  61.             pthread_create(&a_thread, NULL, func1, NULL); 
  62.             // 信號量 + 1,進而觸發 func1 的任務 
  63.             sem_post(&semDownload); 
  64.             // 總線程數 + 1 
  65.             g_phreadNum++; 
  66.             break; 
  67.         case 2: 
  68.             pthread_create(&b_thread, NULL, func2, NULL); 
  69.             sem_post(&semDownload); 
  70.             g_phreadNum++; 
  71.             break; 
  72.         case 3: 
  73.             pthread_create(&c_thread, NULL, func3, NULL); 
  74.             sem_post(&semDownload); 
  75.             g_phreadNum++; 
  76.             break; 
  77.         default
  78.             printf("!!! error taskTypeId %d !!!\n", taskTypeId); 
  79.             break; 
  80.         } 
  81.     } 
  82.     // 銷毀信號量 
  83.     sem_destroy(&semDownload); 
  84.     return 0; 

上述例子中,采用了 pthread_join() 的方式,即子線程合入主線程,主線程阻塞等待子線程結束,然后回收子線程資源。而線程加入還有另外一種方式:pthread_detach(),即主線程與子線程分離,主線程不用關注子線程什么時候結束,子線程結束后,資源自動回收。

程序運行結果如下:

 

深究 Linux 多線程中的信號量 Semaphore

還要注意一點:pthread.h 非 linux 系統的默認庫, gcc 編譯參數需要手動添加選項:-lpthread、-pthread.

責任編輯:未麗燕 來源: 今日頭條
相關推薦

2009-12-08 12:14:43

2021-09-07 07:53:42

Semaphore 信號量源碼

2010-03-16 17:52:27

Java多線程信號量

2020-11-05 09:59:24

Linux內核信號量

2020-09-25 07:34:40

Linux系統編程信號量

2024-10-29 15:23:45

Python線程安全

2024-04-10 08:16:20

多線程編程Java并發編程

2022-04-13 11:12:43

鴻蒙輕內核信號量模塊操作系統

2021-04-30 00:00:50

Semaphore信號量面試官

2010-07-15 15:32:10

Perl線程

2021-05-31 20:30:55

鴻蒙HarmonyOS應用

2021-08-30 08:03:22

信號量Go

2010-04-21 16:50:31

Unix信號量

2010-04-21 16:25:13

Unix信號量

2010-04-21 16:42:48

Unix信號量

2021-02-03 20:10:29

Linux信號量shell

2025-04-16 08:50:00

信號量隔離線程池隔離并發控制

2021-04-13 09:20:15

鴻蒙HarmonyOS應用開發

2010-04-21 17:10:25

Unix信號量

2025-04-23 11:00:00

Hystrix隔離模式信號量
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 久久国际精品 | 午夜私人影院在线观看 | 国产成人综合网 | 亚洲精品一区二区在线观看 | 精精国产xxxx视频在线野外 | 国产性色视频 | 三级在线观看 | 国产成人99久久亚洲综合精品 | 亚洲福利一区二区 | 亚洲日产精品 | 91免费在线视频 | 国产精品一区二区三区久久久 | 亚洲 欧美 在线 一区 | 亚洲综合精品 | 亚洲欧美日韩精品久久亚洲区 | 超级黄色一级片 | 国产小网站 | 国产精品久久久久aaaa九色 | 天堂综合网久久 | 亚洲综合热 | 精品国产综合 | 欧美在线综合 | 欧美h版| 5060网一级毛片 | 中文字幕一区二区三区精彩视频 | 美日韩一区二区 | 亚洲自拍偷拍免费视频 | 91国内精品 | 国产一级电影在线观看 | 亚洲一区二区三区国产 | 久久精品久久久 | 夜夜草视频 | 国产激情免费视频 | 91精品国产综合久久精品 | 激情国产 | 波多野结衣电影一区 | 成人一区av | 亚洲一区中文字幕在线观看 | 在线一区视频 | 偷拍自拍网址 | 欧美一区二区三区在线观看 |