<dfn id="yhprb"><s id="yhprb"></s></dfn><dfn id="yhprb"><delect id="yhprb"></delect></dfn><dfn id="yhprb"></dfn><dfn id="yhprb"><delect id="yhprb"></delect></dfn><dfn id="yhprb"></dfn><dfn id="yhprb"><s id="yhprb"><strike id="yhprb"></strike></s></dfn><small id="yhprb"></small><dfn id="yhprb"></dfn><small id="yhprb"><delect id="yhprb"></delect></small><small id="yhprb"></small><small id="yhprb"></small> <delect id="yhprb"><strike id="yhprb"></strike></delect><dfn id="yhprb"></dfn><dfn id="yhprb"></dfn><s id="yhprb"><noframes id="yhprb"><small id="yhprb"><dfn id="yhprb"></dfn></small><dfn id="yhprb"><delect id="yhprb"></delect></dfn><small id="yhprb"></small><dfn id="yhprb"><delect id="yhprb"></delect></dfn><dfn id="yhprb"><s id="yhprb"></s></dfn> <small id="yhprb"></small><delect id="yhprb"><strike id="yhprb"></strike></delect><dfn id="yhprb"><s id="yhprb"></s></dfn><dfn id="yhprb"></dfn><dfn id="yhprb"><s id="yhprb"></s></dfn><dfn id="yhprb"><s id="yhprb"><strike id="yhprb"></strike></s></dfn><dfn id="yhprb"><s id="yhprb"></s></dfn>

新聞中心

EEPW首頁(yè) > 嵌入式系統 > 設計應用 > 多線(xiàn)程編程之:實(shí)驗內容——“生產(chǎn)者消費者”實(shí)驗

多線(xiàn)程編程之:實(shí)驗內容——“生產(chǎn)者消費者”實(shí)驗

作者: 時(shí)間:2014-10-17 來(lái)源:網(wǎng)絡(luò ) 收藏

  9.3 實(shí)驗內容——“生產(chǎn)者消費者”實(shí)驗

本文引用地址:http://dyxdggzs.com/article/264054.htm

  1.實(shí)驗目的

  “生產(chǎn)者消費者”問(wèn)題是一個(gè)著(zhù)名的同時(shí)性編程問(wèn)題的集合。通過(guò)學(xué)習經(jīng)典的“生產(chǎn)者消費者”問(wèn)題的實(shí)驗,讀者可以進(jìn)一步熟悉中的編程,并且掌握用信號量處理線(xiàn)程間的同步和互斥問(wèn)題。

  2.實(shí)驗內容

  “生產(chǎn)者—消費者”問(wèn)題描述如下。

  有一個(gè)有限緩沖區和兩個(gè)線(xiàn)程:生產(chǎn)者和消費者。他們分別不停地把產(chǎn)品放入緩沖區和從緩沖區中拿走產(chǎn)品。一個(gè)生產(chǎn)者在緩沖區滿(mǎn)的時(shí)候必須等待,一個(gè)消費者在緩沖區空的時(shí)候也必須等待。另外,因為緩沖區是臨界資源,所以生產(chǎn)者和消費者之間必須互斥執行。它們之間的關(guān)系如圖9.4所示。



  圖9.4 生產(chǎn)者消費者問(wèn)題描述

  這里要求使用有名管道來(lái)模擬有限緩沖區,并且使用信號量來(lái)解決“生產(chǎn)者—消費者”問(wèn)題中的同步和互斥問(wèn)題。

  3.實(shí)驗步驟

  (1)信號量的考慮。

  這里使用3個(gè)信號量,其中兩個(gè)信號量avail和full分別用于解決生產(chǎn)者和消費者線(xiàn)程之間的同步問(wèn)題,mutex是用于這兩個(gè)線(xiàn)程之間的互斥問(wèn)題。其中avail表示有界緩沖區中的空單元數,初始值為N;full表示有界緩沖區中非空單元數,初始值為0;mutex是互斥信號量,初始值為1。

  (2)畫(huà)出。

  本實(shí)驗如圖9.5所示。



  圖9.5 “生產(chǎn)者—消費者”實(shí)驗

  (3)編寫(xiě)代碼

  本實(shí)驗的代碼中采用的有界緩沖區擁有3個(gè)單元,每個(gè)單元為5個(gè)字節。為了盡量體現每個(gè)信號量的意義,在程序中生產(chǎn)過(guò)程和消費過(guò)程是隨機(采取0~5s的隨機時(shí)間間隔)進(jìn)行的,而且生產(chǎn)者的速度比消費者的速度平均快兩倍左右(這種關(guān)系可以相反)。生產(chǎn)者一次生產(chǎn)一個(gè)單元的產(chǎn)品(放入“hello”字符串),消費者一次消費一個(gè)單元的產(chǎn)品。

  /*producer-customer.c*/

  #include

  #include

  #include

  #include

  #include

  #include

  #include

  #include

  #define MYFIFO "myfifo" /* 緩沖區有名管道的名字 */

  #define BUFFER_SIZE 3 /* 緩沖區的單元數 */

  #define UNIT_SIZE 5 /* 每個(gè)單元的大小 */

  #define RUN_TIME 30 /* 運行時(shí)間 */

  #define DELAY_TIME_LEVELS 5.0 /* 周期的最大值 */

  int fd;

  time_t end_time;

  sem_t mutex, full, avail; /* 3個(gè)信號量 */

  /*生產(chǎn)者線(xiàn)程*/

  void *producer(void *arg)

  {

  int real_write;

  int delay_time = 0;

  while(time(NULL) < end_time)

  {

  delay_time = (int)(rand() * DELAY_TIME_LEVELS/(RAND_MAX) / 2.0) + 1;

  sleep(delay_time);

  /*P操作信號量avail和mutex*/

  sem_wait(&avail);

  sem_wait(&mutex);

  printf("nProducer: delay = %dn", delay_time);

  /*生產(chǎn)者寫(xiě)入數據*/

  if ((real_write = write(fd, "hello", UNIT_SIZE)) == -1)

  {

  if(errno == EAGAIN)

  {

  printf("The FIFO has not been read yet.Please try latern");

  }

  }

  else

  {

  printf("Write %d to the FIFOn", real_write);

  }

  /*V操作信號量full和mutex*/

  sem_post(&full);

  sem_post(&mutex);

  }

  pthread_exit(NULL);

  }

  /* 消費者線(xiàn)程*/

  void *customer(void *arg)

  {

  unsigned char read_buffer[UNIT_SIZE];

  int real_read;

  int delay_time;

  while(time(NULL) < end_time)

  {

  delay_time = (int)(rand() * DELAY_TIME_LEVELS/(RAND_MAX)) + 1;

  sleep(delay_time);

  /*P操作信號量full和mutex*/

  sem_wait(&full);

  sem_wait(&mutex);

  memset(read_buffer, 0, UNIT_SIZE);

  printf("nCustomer: delay = %dn", delay_time);

  if ((real_read = read(fd, read_buffer, UNIT_SIZE)) == -1)

  {

  if (errno == EAGAIN)

  {

  printf("No data yetn");

  }

  }

  printf("Read %s from FIFOn", read_buffer);

  /*V操作信號量avail和mutex*/

  sem_post(&avail);

  sem_post(&mutex);

  }

  pthread_exit(NULL);

  }

  int main()

  {

  pthread_t thrd_prd_id,thrd_cst_id;

  pthread_t mon_th_id;

  int ret;

  srand(time(NULL));

  end_time = time(NULL) + RUN_TIME;

  /*創(chuàng )建有名管道*/

  if((mkfifo(MYFIFO, O_CREAT|O_EXCL) < 0) && (errno != EEXIST))

  {

  printf("Cannot create fifon");

  return errno;

  }

  /*打開(kāi)管道*/

  fd = open(MYFIFO, O_RDWR);

  if (fd == -1)

  {

  printf("Open fifo errorn");

  return fd;

  }

  /*初始化互斥信號量為1*/

  ret = sem_init(&mutex, 0, 1);

  /*初始化avail信號量為N*/

  ret += sem_init(&avail, 0, BUFFER_SIZE);

  /*初始化full信號量為0*/

  ret += sem_init(&full, 0, 0);

  if (ret != 0)

  {

  printf("Any semaphore initialization failedn");

  return ret;

  }

  /*創(chuàng )建兩個(gè)線(xiàn)程*/

  ret = pthread_create(&thrd_prd_id, NULL, producer, NULL);

  if (ret != 0)

  {

  printf("Create producer thread errorn");

  return ret;

  }

  ret = pthread_create(&thrd_cst_id, NULL, customer, NULL);

  if(ret != 0)

  {

  printf("Create customer thread errorn");

  return ret;

  }

  pthread_join(thrd_prd_id, NULL);

  pthread_join(thrd_cst_id, NULL);

  close(fd);

  unlink(MYFIFO);

  return 0;

  }

  4.實(shí)驗結果

  運行該程序,得到如下結果:

  $ ./producer_customer

  ……

  Producer: delay = 3

  Write 5 to the FIFO

  Customer: delay = 3

  Read hello from FIFO

  Producer: delay = 1

  Write 5 to the FIFO

  Producer: delay = 2

  Write 5 to the FIFO

  Customer: delay = 4

  Read hello from FIFO

  Customer: delay = 1

  Read hello from FIFO

  Producer: delay = 2

  Write 5 to the FIFO

  ……

linux操作系統文章專(zhuān)題:linux操作系統詳解(linux不再難懂)

linux操作系統文章專(zhuān)題:linux操作系統詳解(linux不再難懂)

linux相關(guān)文章:linux教程




關(guān)鍵詞: 多線(xiàn)程 Linux 流程圖

評論


相關(guān)推薦

技術(shù)專(zhuān)區

關(guān)閉
国产精品自在自线亚洲|国产精品无圣光一区二区|国产日产欧洲无码视频|久久久一本精品99久久K精品66|欧美人与动牲交片免费播放
<dfn id="yhprb"><s id="yhprb"></s></dfn><dfn id="yhprb"><delect id="yhprb"></delect></dfn><dfn id="yhprb"></dfn><dfn id="yhprb"><delect id="yhprb"></delect></dfn><dfn id="yhprb"></dfn><dfn id="yhprb"><s id="yhprb"><strike id="yhprb"></strike></s></dfn><small id="yhprb"></small><dfn id="yhprb"></dfn><small id="yhprb"><delect id="yhprb"></delect></small><small id="yhprb"></small><small id="yhprb"></small> <delect id="yhprb"><strike id="yhprb"></strike></delect><dfn id="yhprb"></dfn><dfn id="yhprb"></dfn><s id="yhprb"><noframes id="yhprb"><small id="yhprb"><dfn id="yhprb"></dfn></small><dfn id="yhprb"><delect id="yhprb"></delect></dfn><small id="yhprb"></small><dfn id="yhprb"><delect id="yhprb"></delect></dfn><dfn id="yhprb"><s id="yhprb"></s></dfn> <small id="yhprb"></small><delect id="yhprb"><strike id="yhprb"></strike></delect><dfn id="yhprb"><s id="yhprb"></s></dfn><dfn id="yhprb"></dfn><dfn id="yhprb"><s id="yhprb"></s></dfn><dfn id="yhprb"><s id="yhprb"><strike id="yhprb"></strike></s></dfn><dfn id="yhprb"><s id="yhprb"></s></dfn>