當(dāng)前位置:首頁 > 嵌入式 > 嵌入式教程
[導(dǎo)讀]“生產(chǎn)者消費者”問題是一個著名的同時性編程問題的集合。通過學(xué)習(xí)經(jīng)典的“生產(chǎn)者消費者”問題的實驗,讀者可以進(jìn)一步熟悉Linux中的多線程編程,并且掌握用信號量處理線程間的同步和互斥問題。

9.3實驗內(nèi)容——“生產(chǎn)者消費者”實驗1.實驗?zāi)康?p>“生產(chǎn)者消費者”問題是一個著名的同時性編程問題的集合。通過學(xué)習(xí)經(jīng)典的“生產(chǎn)者消費者”問題的實驗,讀者可以進(jìn)一步熟悉Linux中的多線程編程,并且掌握用信號量處理線程間的同步和互斥問題。

2.實驗內(nèi)容

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

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

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

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

3.實驗步驟

(1)信號量的考慮。

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

(2)畫出流程圖。

本實驗流程圖如圖9.5所示。

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

(3)編寫代碼

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

/*producer-customer.c*/

#include<stdio.h>

#include<stdlib.h>

#include<unistd.h>

#include<fcntl.h>

#include<pthread.h>

#include<errno.h>

#include<semaphore.h>

#include<sys/ipc.h>

#defineMYFIFO"myfifo"/*緩沖區(qū)有名管道的名字*/

#defineBUFFER_SIZE3/*緩沖區(qū)的單元數(shù)*/

#defineUNIT_SIZE5/*每個單元的大小*/

#defineRUN_TIME30/*運行時間*/

#defineDELAY_TIME_LEVELS5.0/*周期的最大值*/

intfd;

time_tend_time;

sem_tmutex,full,avail;/*3個信號量*/

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

void*producer(void*arg)

{

intreal_write;

intdelay_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)者寫入數(shù)據(jù)*/

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

{

if(errno==EAGAIN)

{

printf("TheFIFOhasnotbeenreadyet.Pleasetrylatern");

}

}

else

{

printf("Write%dtotheFIFOn",real_write);

}

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

sem_post(&full);

sem_post(&mutex);

}

pthread_exit(NULL);

}

/*消費者線程*/

void*customer(void*arg)

{

unsignedcharread_buffer[UNIT_SIZE];

intreal_read;

intdelay_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("Nodatayetn");

}

}

printf("Read%sfromFIFOn",read_buffer);

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

sem_post(&avail);

sem_post(&mutex);

}

pthread_exit(NULL);

}

intmain()

{

pthread_tthrd_prd_id,thrd_cst_id;

pthread_tmon_th_id;

intret;

srand(time(NULL));

end_time=time(NULL)+RUN_TIME;

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

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

{

printf("Cannotcreatefifon");

returnerrno;

}

/*打開管道*/

fd=open(MYFIFO,O_RDWR);

if(fd==-1)

{

printf("Openfifoerrorn");

returnfd;

}

/*初始化互斥信號量為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("Anysemaphoreinitializationfailedn");

returnret;

}

/*創(chuàng)建兩個線程*/

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

if(ret!=0)

{

printf("Createproducerthreaderrorn");

returnret;

}

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

if(ret!=0)

{

printf("Createcustomerthreaderrorn");

returnret;

}

pthread_join(thrd_prd_id,NULL);

pthread_join(thrd_cst_id,NULL);

close(fd);

unlink(MYFIFO);

return0;

}

4.實驗結(jié)果

運行該程序,得到如下結(jié)果:

$./producer_customer

……

Producer:delay=3

Write5totheFIFO

Customer:delay=3

ReadhellofromFIFO

Producer:delay=1

Write5totheFIFO

Producer:delay=2

Write5totheFIFO

Customer:delay=4

ReadhellofromFIFO

Customer:delay=1

ReadhellofromFIFO

Producer:delay=2

Write5totheFIFO

……

本站聲明: 本文章由作者或相關(guān)機(jī)構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點,本站亦不保證或承諾內(nèi)容真實性等。需要轉(zhuǎn)載請聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請及時聯(lián)系本站刪除。
換一批
延伸閱讀

9月2日消息,不造車的華為或?qū)⒋呱龈蟮莫毥谦F公司,隨著阿維塔和賽力斯的入局,華為引望愈發(fā)顯得引人矚目。

關(guān)鍵字: 阿維塔 塞力斯 華為

加利福尼亞州圣克拉拉縣2024年8月30日 /美通社/ -- 數(shù)字化轉(zhuǎn)型技術(shù)解決方案公司Trianz今天宣布,該公司與Amazon Web Services (AWS)簽訂了...

關(guān)鍵字: AWS AN BSP 數(shù)字化

倫敦2024年8月29日 /美通社/ -- 英國汽車技術(shù)公司SODA.Auto推出其旗艦產(chǎn)品SODA V,這是全球首款涵蓋汽車工程師從創(chuàng)意到認(rèn)證的所有需求的工具,可用于創(chuàng)建軟件定義汽車。 SODA V工具的開發(fā)耗時1.5...

關(guān)鍵字: 汽車 人工智能 智能驅(qū)動 BSP

北京2024年8月28日 /美通社/ -- 越來越多用戶希望企業(yè)業(yè)務(wù)能7×24不間斷運行,同時企業(yè)卻面臨越來越多業(yè)務(wù)中斷的風(fēng)險,如企業(yè)系統(tǒng)復(fù)雜性的增加,頻繁的功能更新和發(fā)布等。如何確保業(yè)務(wù)連續(xù)性,提升韌性,成...

關(guān)鍵字: 亞馬遜 解密 控制平面 BSP

8月30日消息,據(jù)媒體報道,騰訊和網(wǎng)易近期正在縮減他們對日本游戲市場的投資。

關(guān)鍵字: 騰訊 編碼器 CPU

8月28日消息,今天上午,2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會開幕式在貴陽舉行,華為董事、質(zhì)量流程IT總裁陶景文發(fā)表了演講。

關(guān)鍵字: 華為 12nm EDA 半導(dǎo)體

8月28日消息,在2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會上,華為常務(wù)董事、華為云CEO張平安發(fā)表演講稱,數(shù)字世界的話語權(quán)最終是由生態(tài)的繁榮決定的。

關(guān)鍵字: 華為 12nm 手機(jī) 衛(wèi)星通信

要點: 有效應(yīng)對環(huán)境變化,經(jīng)營業(yè)績穩(wěn)中有升 落實提質(zhì)增效舉措,毛利潤率延續(xù)升勢 戰(zhàn)略布局成效顯著,戰(zhàn)新業(yè)務(wù)引領(lǐng)增長 以科技創(chuàng)新為引領(lǐng),提升企業(yè)核心競爭力 堅持高質(zhì)量發(fā)展策略,塑強(qiáng)核心競爭優(yōu)勢...

關(guān)鍵字: 通信 BSP 電信運營商 數(shù)字經(jīng)濟(jì)

北京2024年8月27日 /美通社/ -- 8月21日,由中央廣播電視總臺與中國電影電視技術(shù)學(xué)會聯(lián)合牽頭組建的NVI技術(shù)創(chuàng)新聯(lián)盟在BIRTV2024超高清全產(chǎn)業(yè)鏈發(fā)展研討會上宣布正式成立。 活動現(xiàn)場 NVI技術(shù)創(chuàng)新聯(lián)...

關(guān)鍵字: VI 傳輸協(xié)議 音頻 BSP

北京2024年8月27日 /美通社/ -- 在8月23日舉辦的2024年長三角生態(tài)綠色一體化發(fā)展示范區(qū)聯(lián)合招商會上,軟通動力信息技術(shù)(集團(tuán))股份有限公司(以下簡稱"軟通動力")與長三角投資(上海)有限...

關(guān)鍵字: BSP 信息技術(shù)
關(guān)閉
關(guān)閉