亚洲综合原千岁中文字幕_国产精品99久久久久久久vr_无码人妻aⅴ一区二区三区浪潮_成人h动漫精品一区二区三

主頁 > 知識庫 > linux多線程編程詳解教程(線程通過信號量實現通信代碼)

linux多線程編程詳解教程(線程通過信號量實現通信代碼)

熱門標簽:百度地圖標注錯了有責任嗎 江蘇云電銷機器人公司 華鋒e路航港口地圖標注 河南信譽好的不封卡電話外呼系統 打電話機器人接我是他的秘書 地圖標注員都是年輕人 揭陽智能電話機器人推薦 如果做線上地圖標注 客服外呼系統怎么樣

線程分類

線程按照其調度者可以分為用戶級線程和核心級線程兩種。

(1)用戶級線程
用戶級線程主要解決的是上下文切換的問題,它的調度算法和調度過程全部由用戶自行選擇決定,在運行時不需要特定的內核支持。在這里,操作系統往往會提供一個用戶空間的線程庫,該線程庫提供了線程的創建、調度、撤銷等功能,而內核仍然僅對進程進行管理。如果一個進程中的某一個線程調用了一個阻塞的系統調用,那么該進程包括該進程中的其他所有線程也同時被阻塞。這種用戶級線程的主要缺點是在一個進程中的多個線程的調度中無法發揮多處理器的優勢。

(2)核心級線程
這種線程允許不同進程中的線程按照同一相對優先調度方法進行調度,這樣就可以發揮多處理器的并發優勢。
現在大多數系統都采用用戶級線程與核心級線程并存的方法。一個用戶級線程可以對應一個或幾個核心級線程,也就是“一對一”或“多對一”模型。這樣既可滿足多處理機系統的需要,也可以最大限度地減少調度開銷。

Linux的線程實現是在核外進行的,核內提供的是創建進程的接口do_fork()。內核提供了兩個系統調用clone()和fork(),最終都用不同的參數調用do_fork()核內API。當然,要想實現線程,沒有核心對多進程(其實是輕量級進程)共享數據段的支持是不行的,因此,do_fork()提供了很多參數,包括CLONE_VM(共享內存空間)、CLONE_FS(共享文件系統信息)、 CLONE_FILES(共享文件描述符表)、CLONE_SIGHAND(共享信號句柄表)和CLONE_PID(共享進程ID,僅對核內進程,即0號進程有效)。當使用fork系統調用時,內核調用do_fork()不使用任何共享屬性,進程擁有獨立的運行環境,而使用 pthread_create()來創建線程時,則最終設置了所有這些屬性來調用__clone(),而這些參數又全部傳給核內的do_fork(),從而創建的“進程”擁有共享的運行環境,只有棧是獨立的,由__clone()傳入。

Linux線程在核內是以輕量級進程的形式存在的,擁有獨立的進程表項,而所有的創建、同步、刪除等操作都在核外pthread庫中進行。pthread 庫使用一個管理線程(__pthread_manager(),每個進程獨立且唯一)來管理線程的創建和終止,為線程分配線程ID,發送線程相關的信號(比如Cancel),而主線程(pthread_create())的調用者則通過管道將請求信息傳給管理線程。

主要函數說明

1.線程的創建和退出

pthread_create 線程創建函數
int pthread_create (pthread_t * thread_id,__const pthread_attr_t * __attr,void *(*__start_routine) (void *),void *__restrict __arg);

線程創建函數第一個參數為指向線程標識符的指針,第二個參數用來設置線程屬性,第三個參數是線程運行函數的起始地址,最后一個參數是運行函數的參數。這里,我們的函數thread 不需要參數,所以最后一個參數設為空指針。第二個參數我們也設為空指針,這樣將生成默認屬性的線程。當創建線程成功時,函數返回0,若不為0 則說明創建線程失敗,常見的錯誤返回代碼為EAGAIN 和EINVAL。前者表示系統限制創建新的線程,例如線程數目過多了;后者表示第二個參數代表的線程屬性值非法。創建線程成功后,新創建的線程則運行參數三和參數四確定的函數,原來的線程則繼續運行下一行代碼。

pthread_join 函數,來等待一個線程的結束。
函數原型為:int pthread_join (pthread_t __th, void **__thread_return)
第一個參數為被等待的線程標識符,第二個參數為一個用戶定義的指針,它可以用來存儲被等待線程的返回值。這個函數是一個線程阻塞的函數,調用它的函數將一直等待到被等待的線程結束為止,當函數返回時,被等待線程的資源被收回。線程只能被一個線程等待終止,并且應處于joinable狀態(非detached)。

pthread_exit 函數
一個線程的結束有兩種途徑,一種是線程運行的函數結束了,調用它的線程也就結束了;
另一種方式是通過函數pthread_exit 來實現。它的函數原型為:void pthread_exit (void *__retval)唯一的參數是函數的返回代碼,只要pthread_join 中的第二個參數thread_return 不是NULL,這個值將被傳遞給thread_return。最后要說明的是,一個線程不能被多個線程等待,否則第一個接收到信號的線程成功返回,其余調用pthread_join 的線程則返回錯誤代碼ESRCH。

2.線程屬性

pthread_create函數的第二個參數線程的屬性。將該值設為NULL,也就是采用默認屬性,線程的多項屬性都是可以更改的。這些屬性主要包括綁定屬性、分離屬性、堆棧地址、堆棧大小、優先級。其中系統默認的屬性為非綁定、非分離、缺省1M 的堆棧、與父進程同樣級別的優先級。下面首先對綁定屬性和分離屬性的基本概念進行講解。

綁定屬性:Linux中采用“一對一”的線程機制,也就是一個用戶線程對應一個內核線程。綁定屬性就是指一個用戶線程固定地分配給一個內核線程,因為CPU時間片的調度是面向內核線程 (也就是輕量級進程)的,因此具有綁定屬性的線程可以保證在需要的時候總有一個內核線程與之對應。而與之相對的非綁定屬性就是指用戶線程和內核線程的關系不是始終固定的,而是由系統來控制分配的。

分離屬性:分離屬性是用來決定一個線程以什么樣的方式來終止自己。在非分離情況下,當一個線程結束時,它所占用的系統資源并沒有被釋放,也就是沒有真正的終止。只有當pthread_join()函數返回時,創建的線程才能釋放自己占用的系統資源。而在分離屬性情況下,一個線程結束時立即釋放它所占有的系統資源。
這里要注意的一點是,如果設置一個線程的分離屬性,而這個線程運行又非常快,那么它很可能在pthread_create 函數返回之前就終止了,它終止以后就可能將線程號和系統資源移交給其他的線程使用,這時調用pthread_create 的線程就得到了錯誤的線程號。

設置綁定屬性:

int pthread_attr_init(pthread_attr_t *attr)
int pthread_attr_setscope(pthread_attr_t *attr, int scope)
int pthread_attr_getscope(pthread_attr_t *tattr, int *scope)
scope:PTHREAD_SCOPE_SYSTEM:綁定,此線程與系統中所有的線程競爭 PTHREAD_SCOPE_PROCESS:非綁定,此線程與進程中的其他線程競爭

設置分離屬性:

int pthread_attr_setdetachstate(pthread_attr_t *attr, int detachstate)
int pthread_attr_getdetachstate(const pthread_attr_t *tattr,int *detachstate)
detachstate PTHREAD_CREATE_DETACHED:分離 PTHREAD _CREATE_JOINABLE:非分離

設置調度策略:

int pthread_attr_setschedpolicy(pthread_attr_t * tattr, int policy)
int pthread_attr_getschedpolicy(pthread_attr_t * tattr, int *policy)
policy SCHED_FIFO:先入先出 SCHED_RR:循環 SCHED_OTHER:實現定義的方法

設置優先級:

int pthread_attr_setschedparam (pthread_attr_t *attr, struct sched_param *param)
int pthread_attr_getschedparam (pthread_attr_t *attr, struct sched_param *param)

3.線程訪問控制

1)互斥鎖(mutex)
通過鎖機制實現線程間的同步。同一時刻只允許一個線程執行一個關鍵部分的代碼。

1 int pthread_mutex_init(pthread_mutex_t *mutex,const pthread_mutex_attr_t *mutexattr);
2 int pthread_mutex_lock(pthread_mutex_t *mutex);
3 int pthread_mutex_unlock(pthread_mutex_t *mutex);
4 int pthread_mutex_destroy(pthread_mutex_t *mutex);

(1)先初始化鎖init()或靜態賦值pthread_mutex_t mutex=PTHREAD_MUTEX_INITIALIER
(2)加鎖,lock,trylock,lock阻塞等待鎖,trylock立即返回EBUSY
(3)解鎖,unlock需滿足是加鎖狀態,且由加鎖線程解鎖
(4)清除鎖,destroy(此時鎖必需unlock,否則返回EBUSY)

mutex 分為遞歸(recursive) 和非遞歸(non-recursive)兩種,這是POSIX 的叫法,另外的名字是可重入(Reentrant) 與非可重入。這兩種mutex 作為線程間(inter-thread) 的同步工具時沒有區別,它們的惟一區別在于:同一個線程可以重復對recursive mutex 加鎖,但是不能重復對non-recursive mutex 加鎖。
首選非遞歸mutex,絕對不是為了性能,而是為了體現設計意圖。non-recursive 和recursive 的性能差別其實不大,因為少用一個計數器,前者略快一點點而已。在同一個線程里多次對non-recursive mutex 加鎖會立刻導致死鎖,我認為這是它的優點,能幫助我們思考代碼對鎖的期求,并且及早(在編碼階段)發現問題。毫無疑問recursive mutex 使用起來要方便一些,因為不用考慮一個線程會自己把自己給鎖死了,我猜這也是Java 和Windows 默認提供recursive mutex 的原因。(Java 語言自帶的intrinsic lock 是可重入的,它的concurrent 庫里提供ReentrantLock,Windows的CRITICAL_SECTION 也是可重入的。似乎它們都不提供輕量級的non-recursive mutex。)

2)條件變量(cond)
利用線程間共享的全局變量進行同步的一種機制。

1 int pthread_cond_init(pthread_cond_t *cond,pthread_condattr_t *cond_attr);
2 int pthread_cond_wait(pthread_cond_t *cond,pthread_mutex_t *mutex);
3 int pthread_cond_timedwait(pthread_cond_t *cond,pthread_mutex_t *mutex,const timespec *abstime);
4 int pthread_cond_destroy(pthread_cond_t *cond); 
5 int pthread_cond_signal(pthread_cond_t *cond);
6 int pthread_cond_broadcast(pthread_cond_t *cond);  //解除所有線程的阻塞


(1)初始化. init()或者pthread_cond_t cond=PTHREAD_COND_INITIALIER;屬性置為NULL
(2)等待條件成立. pthread_cond_wait,pthread_cond_timedwait.
wait()釋放鎖,并阻塞等待條件變量為真
timedwait()設置等待時間,仍未signal,返回ETIMEOUT(加鎖保證只有一個線程wait)
(3)激活條件變量:pthread_cond_signal,pthread_cond_broadcast(激活所有等待線程)
(4)清除條件變量:destroy; 無線程等待,否則返回EBUSY

復制代碼 代碼如下:

int pthread_cond_wait(pthread_cond_t *cond, pthread_mutex_t *mutex);
int pthread_cond_timedwait(pthread_cond_t *cond, pthread_mutex_t *mutex, const struct timespec *abstime);

這兩個函數一定要在mutex的鎖定區域內使用。

調用 pthread_cond_signal() 釋放被條件阻塞的線程時,如果沒有任何線程基于條件變量阻塞,則調用pthread_cond_signal()不起作用。而對于 Windows,當調用 SetEvent 觸發 Auto-reset 的 Event 條件時,如果沒有被條件阻塞的線程,那么此函數仍然起作用,條件變量會處在觸發狀態。

Linux下生產者消費者問題(使用互斥鎖和條件變量):

復制代碼 代碼如下:

#include stdio.h>
#include stdlib.h>
#include time.h>
#include "pthread.h"

#define BUFFER_SIZE 16

struct prodcons 

int buffer[BUFFER_SIZE]; 
pthread_mutex_t lock;  //mutex ensuring exclusive access to buffer
int readpos,writepos;  //position for reading and writing
pthread_cond_t notempty;  //signal when buffer is not empty
pthread_cond_t notfull;  //signal when buffer is not full
}; 

//initialize a buffer
void init(struct prodcons* b) 

pthread_mutex_init(b->lock,NULL); 
pthread_cond_init(b->notempty,NULL); 
pthread_cond_init(b->notfull,NULL); 
b->readpos = 0; 
b->writepos = 0; 

//store an integer in the buffer
void put(struct prodcons* b, int data) 

pthread_mutex_lock(b->lock); 
//wait until buffer is not full
while((b->writepos+1)%BUFFER_SIZE == b->readpos) 

printf("wait for not full\n"); 
pthread_cond_wait(b->notfull,b->lock); 
}
b->buffer[b->writepos] = data; 
b->writepos++;
b->writepos %= BUFFER_SIZE;
pthread_cond_signal(b->notempty); //signal buffer is not empty
pthread_mutex_unlock(b->lock); 
}

//read and remove an integer from the buffer
int get(struct prodcons* b) 

int data; 
pthread_mutex_lock(b->lock); 
//wait until buffer is not empty
while(b->writepos == b->readpos) 

printf("wait for not empty\n"); 
pthread_cond_wait(b->notempty,b->lock); 
}
data=b->buffer[b->readpos]; 
b->readpos++;
b->readpos %= BUFFER_SIZE;
pthread_cond_signal(b->notfull);  //signal buffer is not full
pthread_mutex_unlock(b->lock); 
return data;
}

#define OVER -1

struct prodcons buffer; 

void * producer(void * data) 

int n; 
for(n=0; n50; ++n) 
{
printf("put-->%d\n",n); 
put(buffer,n); 

put(buffer,OVER); 
printf("producer stopped\n"); 
return NULL; 

void * consumer(void * data) 

int n; 
while(1) 

int d = get(buffer); 
if(d == OVER) break; 
printf("get-->%d\n",d); 
}
printf("consumer stopped\n"); 
return NULL; 

int main() 

pthread_t tha,thb; 
void * retval; 

init(buffer); 
pthread_creare(tha,NULL,producer,0); 
pthread_creare(thb,NULL,consumer,0); 

pthread_join(tha,retval); 
pthread_join(thb,retval); 

return 0; 
}

3)信號量
如同進程一樣,線程也可以通過信號量來實現通信,雖然是輕量級的。

信號量函數的名字都以"sem_"打頭。線程使用的基本信號量函數有四個。

復制代碼 代碼如下:

#include semaphore.h>
int sem_init(sem_t *sem , int pshared, unsigned int value);

這是對由sem指定的信號量進行初始化,設置好它的共享選項(linux只支持為0,即表示它是當前進程的局部信號量),然后給它一個初始值VALUE。

兩個原子操作函數:這兩個函數都要用一個由sem_init調用初始化的信號量對象的指針做參數。

復制代碼 代碼如下:

int sem_wait(sem_t *sem); //給信號量減1,對一個值為0的信號量調用sem_wait,這個函數將會等待直到有其它線程使它不再是0為止。
int sem_post(sem_t *sem); //給信號量的值加1

int sem_destroy(sem_t *sem);

這個函數的作用是再我們用完信號量后都它進行清理。歸還自己占有的一切資源。

用信號量實現生產者消費者:

這里使用4個信號量,其中兩個信號量occupied和empty分別用于解決生產者和消費者線程之間的同步問題,pmut用于多個生產者之間互斥問題,cmut是用于多個消費者之間互斥問題。其中empty初始化為N(有界緩區的空間元數),occupied初始化為0,pmut和cmut初始化為1。

參考代碼:

復制代碼 代碼如下:

#define BSIZE 64

typedef struct
{
char buf[BSIZE];
sem_t occupied;
sem_t empty;
int nextin;
int nextout;
sem_t pmut;
sem_t cmut;
}buffer_t;

buffer_t buffer;

void init(buffer_t * b)
{
sem_init(b->occupied, 0, 0);
sem_init(b->empty,0, BSIZE);
sem_init(b->pmut, 0, 1);
sem_init(b->cmut, 0, 1);
b->nextin = b->nextout = 0;
}

void producer(buffer_t *b, char item)
{
sem_wait(b->empty);
sem_wait(b->pmut);
b->buf[b->nextin] = item;
b->nextin++;
b->nextin %= BSIZE;
sem_post(b->pmut);
sem_post(b->occupied);
}

char consumer(buffer_t *b)
{
char item;
sem_wait(b->occupied);
sem_wait(b->cmut);
item = b->buf[b->nextout];
b->nextout++;
b->nextout %= BSIZE;
sem_post(b->cmut);
sem_post(b->empty);
return item;
}

您可能感興趣的文章:
  • golang 監聽服務的信號,實現平滑啟動,linux信號說明詳解
  • Linux進程間通信--使用信號
  • 詳解Linux進程間通信——使用信號量
  • 詳解Linux多線程使用信號量同步
  • Linux下的信號詳解及捕捉信號
  • linux下基于C語言的信號編程實例
  • Linux線程同步之信號C語言實例
  • Linux下semop等待信號時出現Interrupted System Call錯誤(EINTR)解決方法
  • 淺談Linux信號機制

標簽:馬鞍山 淘寶邀評 婁底 許昌 赤峰 邵陽 巴彥淖爾 金昌

巨人網絡通訊聲明:本文標題《linux多線程編程詳解教程(線程通過信號量實現通信代碼)》,本文關鍵詞  linux,多,線程,編程,詳解,;如發現本文內容存在版權問題,煩請提供相關信息告之我們,我們將及時溝通與處理。本站內容系統采集于網絡,涉及言論、版權與本站無關。
  • 相關文章
  • 下面列出與本文章《linux多線程編程詳解教程(線程通過信號量實現通信代碼)》相關的同類信息!
  • 本頁收集關于linux多線程編程詳解教程(線程通過信號量實現通信代碼)的相關信息資訊供網民參考!
  • 推薦文章
    青青青草影院| 黄色福利片| 麻豆午夜视频| 亚欧乱色一区二区三区| 国产a一级| 青青久久精品| 国产亚洲免费观看| 韩国毛片| 国产高清在线精品一区a| 在线观看成人网| a级黄色毛片免费播放视频| 你懂的国产精品| 精品视频在线观看一区二区| 成人免费观看男女羞羞视频| 亚洲第一页色| 日本久久久久久久 97久久精品一区二区三区 狠狠色噜噜狠狠狠狠97 日日干综合 五月天婷婷在线观看高清 九色福利视频 | 日韩免费在线| 精品国产一区二区三区久久久蜜臀| 久久国产精品只做精品| 精品国产一区二区三区免费 | 精品视频在线看| 黄视频网站免费| 欧美另类videosbestsex视频| 一级片片| 精品视频在线观看一区二区| 国产一区二区精品久久91| 欧美激情一区二区三区视频| 四虎影视久久| 日韩在线观看免费完整版视频| 精品国产一级毛片| 日韩中文字幕一区二区不卡| 免费国产一级特黄aa大片在线| 黄视频网站在线观看| 青青青草影院| 国产网站在线| 国产网站免费观看| 91麻豆tv| 国产一区二区精品尤物| 国产国语对白一级毛片| 韩国三级视频网站| 韩国毛片免费大片| 日韩中文字幕在线观看视频| 免费毛片基地| 麻豆污视频| 国产成人啪精品| 精品国产一区二区三区国产馆| 午夜欧美成人香蕉剧场| 成人高清视频免费观看| 欧美爱爱网| 九九干| 一级毛片视频播放| 精品久久久久久中文字幕2017| 国产成+人+综合+亚洲不卡| 午夜在线影院| 成人免费一级毛片在线播放视频| 欧美激情一区二区三区视频高清 | 色综合久久天天综线观看| 日韩专区在线播放| 99久久精品费精品国产一区二区| 欧美激情在线精品video| 黄色免费三级| 黄视频网站在线观看| 国产网站免费观看| 可以免费看毛片的网站| 四虎久久影院| 久久成人亚洲| 美女免费精品高清毛片在线视 | 精品视频免费在线| 日日夜夜婷婷| 日本特黄一级| 国产91精品露脸国语对白| 国产91精品一区二区| 免费一级片在线观看| 日日夜夜婷婷| 日日爽天天| 国产麻豆精品免费密入口| 精品国产亚洲一区二区三区| 一级毛片视频免费| 99久久精品费精品国产一区二区| 国产网站在线| 国产不卡福利| 精品国产亚一区二区三区| 久久99中文字幕| 久久成人亚洲| 国产精品1024在线永久免费| 日韩综合| 黄色短视频网站| 成人高清视频在线观看| 青青青草影院 | 成人高清护士在线播放| 国产伦精品一区二区三区无广告 | 欧美一级视频免费| 天天色成人| 精品视频免费在线| 亚欧成人乱码一区二区| 精品毛片视频| 99久久精品国产麻豆| 国产一区国产二区国产三区| 久草免费在线色站| 成人a大片在线观看| 青青青草影院 | 成人a大片高清在线观看| 99色播| 韩国三级视频网站| 国产精品自拍在线| 精品视频在线看 | 天堂网中文在线| 麻豆系列 在线视频| 日日爽天天| 国产不卡高清| 日日夜夜婷婷| 欧美另类videosbestsex视频| 精品视频在线看 | 日韩欧美一二三区| 精品在线视频播放| 欧美日本免费| 国产不卡在线观看视频| 国产福利免费视频| 午夜精品国产自在现线拍| 精品毛片视频| 高清一级片| 色综合久久天天综线观看| 久久国产精品自由自在| 日韩中文字幕在线播放| 天天做日日爱| 日韩欧美一二三区| 美女免费毛片| 欧美激情影院| 色综合久久天天综线观看| 精品国产一区二区三区精东影业| 欧美一区二区三区在线观看| 高清一级毛片一本到免费观看| 久久国产影院| 国产麻豆精品| 国产精品自拍在线| 午夜在线亚洲| 亚洲www美色| 99久久视频| 国产视频久久久久| 九九九网站| 久久国产精品只做精品| 香蕉视频三级| 日本伦理片网站| 欧美激情一区二区三区在线| 美女免费黄网站| 四虎久久精品国产| 深夜做爰性大片中文| 国产成人精品影视| 亚久久伊人精品青青草原2020| 亚欧视频在线| 精品国产一级毛片| 韩国三级视频在线观看| 欧美激情一区二区三区在线| 国产极品白嫩美女在线观看看| 成人免费观看男女羞羞视频| 日韩免费在线观看视频| 可以免费看污视频的网站| 精品视频在线观看一区二区三区| 免费国产在线视频| 深夜做爰性大片中文| 国产91精品一区二区| 二级片在线观看| 黄视频网站免费看| 国产精品自拍一区| 成人免费高清视频| 国产麻豆精品高清在线播放| 黄视频网站在线看| 成人高清视频在线观看| 可以免费看毛片的网站| 欧美夜夜骑 青草视频在线观看完整版 久久精品99无色码中文字幕 欧美日韩一区二区在线观看视频 欧美中文字幕在线视频 www.99精品 香蕉视频久久 | 中文字幕一区二区三区 精品| 午夜在线亚洲| 日韩av成人| 国产视频一区二区在线观看 | 毛片电影网| 亚洲精品影院| 亚洲天堂免费观看| 欧美国产日韩一区二区三区| 国产一区二区精品久久91| 国产网站在线| 午夜精品国产自在现线拍| 国产一区国产二区国产三区| 青青青草影院| 九九久久国产精品| 国产国语对白一级毛片| 一级片片| 国产麻豆精品视频| 日韩中文字幕在线播放| 99热视热频这里只有精品| 亚洲 欧美 成人日韩| 成人免费观看的视频黄页| 九九久久99| 国产a视频| 欧美激情一区二区三区视频| 韩国三级一区| 精品国产一区二区三区免费 | 免费一级片在线| 国产网站免费视频| 国产一区二区精品| 久久福利影视|