哲學家就餐問題 C語言實現


場景:

 

原版的故事里有五個哲學家(不過我們寫的程序可以有N個哲學家),這些哲學家們只做兩件事--思考和吃飯,他們思考的時候不需要任何共享資源,但是吃飯的時候就必須使用餐具,而餐桌上的餐具是有限的,原版的故事里,餐具是叉子,吃飯的時候要用兩把叉子把面條從碗里撈出來。很顯然把叉子換成筷子會更合理,所以:一個哲學家需要兩根筷子才能吃飯。

現在引入問題的關鍵:這些哲學家很窮,只買得起五根筷子。他們坐成一圈,兩個人的中間放一根筷子。哲學家吃飯的時候必須同時得到左手邊和右手邊的筷子。如果他身邊的任何一位正在使用筷子,那他只有等着。

假設哲學家的編號是A、B、C、D、E,筷子編號是1、2、3、4、5,哲學家和筷子圍成一圈如下圖所示:

 
 

 

  1. #include <stdio.h>  
  2. #include <stdlib.h>  
  3. #include <memory.h>  
  4. #include <pthread.h>  
  5. #include <errno.h>  
  6. #include <math.h>  
  7. //筷子作為mutex  
  8. pthread_mutex_t chopstick[6] ;  
  9. void *eat_think(void *arg)  
  10. {  
  11.     char phi = *(char *)arg;  
  12.     int left,right; //左右筷子的編號  
  13.     switch (phi){  
  14.         case 'A':  
  15.             left = 5;  
  16.             right = 1;  
  17.             break;  
  18.         case 'B':  
  19.             left = 1;  
  20.             right = 2;  
  21.             break;  
  22.         case 'C':  
  23.             left = 2;  
  24.             right = 3;  
  25.             break;  
  26.         case 'D':  
  27.             left = 3;  
  28.             right = 4;  
  29.             break;  
  30.         case 'E':  
  31.             left = 4;  
  32.             right = 5;  
  33.             break;  
  34.     }  
  35.   
  36.     int i;  
  37.     for(;;){  
  38.         usleep(3); //思考  
  39.         pthread_mutex_lock(&chopstick[left]); //拿起左手的筷子  
  40.         printf("Philosopher %c fetches chopstick %d\n", phi, left);  
  41.         if (pthread_mutex_trylock(&chopstick[right]) == EBUSY){ //拿起右手的筷子     
  42.             pthread_mutex_unlock(&chopstick[left]); //如果右邊筷子被拿走放下左手的筷子  
  43.             continue;  
  44.         }  
  45.           
  46.     //  pthread_mutex_lock(&chopstick[right]); //拿起右手的筷子,如果想觀察死鎖,把上一句if注釋掉,再把這一句的注釋去掉  
  47.         printf("Philosopher %c fetches chopstick %d\n", phi, right);  
  48.         printf("Philosopher %c is eating.\n",phi);  
  49.         usleep(3); //吃飯  
  50.         pthread_mutex_unlock(&chopstick[left]); //放下左手的筷子  
  51.         printf("Philosopher %c release chopstick %d\n", phi, left);  
  52.         pthread_mutex_unlock(&chopstick[right]); //放下左手的筷子  
  53.         printf("Philosopher %c release chopstick %d\n", phi, right);  
  54.   
  55.     }  
  56. }  
  57. int main(){  
  58.     pthread_t A,B,C,D,E; //5個哲學家  
  59.   
  60.     int i;  
  61.     for (i = 0; i < 5; i++)  
  62.         pthread_mutex_init(&chopstick[i],NULL);  
  63.     pthread_create(&A,NULL, eat_think, "A");  
  64.     pthread_create(&B,NULL, eat_think, "B");  
  65.     pthread_create(&C,NULL, eat_think, "C");  
  66.     pthread_create(&D,NULL, eat_think, "D");  
  67.     pthread_create(&E,NULL, eat_think, "E");  
  68.   
  69.     pthread_join(A,NULL);  
  70.     pthread_join(B,NULL);  
  71.     pthread_join(C,NULL);  
  72.     pthread_join(D,NULL);  
  73.     pthread_join(E,NULL);  
  74.     return 0;  
  75. }  

注意:

 

1. gcc編譯時,加上-lphread選項,例:gcc eating.c -lpthread

2.pthread_create的第四個參數是void *類型,我一開始傳一個char類型,即'A',就會發生段錯誤。但改成char *,即“A”就沒問題了。

3.usleep是毫秒級的阻塞


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM