linux多線程編程(四)
linux線程分為兩類:一是核心級支持線程,二是用戶級的線程。一般都為用戶級的線程。
一、多線程的幾個常見函數(shù)
要創(chuàng)建多線程必須加載pthread.h文件,庫文件pthread。線程的標(biāo)識符pthread_t在頭文件/usr/include/bits/pthreadtypes.h中定義:typedef unsigned long int pthread_t
1.創(chuàng)建線程:
int pthread_create(pthread_t *restrict thread,
const pthread_attr_t *restrict attr,
void *(*start_routine)(void*), void *restrict arg);
參數(shù):
thread輸出線程id
attr 線程屬性, 默認(rèn)NULL
start_routine線程執(zhí)行函數(shù)
arg線程執(zhí)行參數(shù)
note:函數(shù)成功返回0 否則返回錯誤碼
2.等待指定線程結(jié)束:
int pthread_join(pthread_t thread,void **value_ptr);
參數(shù):
thread一個有效的線程id
value_ptr 接收線程返回值的指針
note:調(diào)用此函數(shù)的線程在指定的線程退出前將處于掛起狀態(tài)或出現(xiàn)錯誤而直接返回,如果value_ptr非NULL則value_ptr指向線程返回值的指針,函數(shù)成功后指定的線程使用的資源將被釋放。
3.退出線程:
int pthread_exit(void * value_ptr);
參數(shù):
value_ptr 線程返回值指針
note: ptrhead_exit()退出調(diào)用此函數(shù)的線程并釋放該線程占用的資源。
4.獲取當(dāng)前線程id:
pthread_t pthread_self(void);
參數(shù):
note:返回當(dāng)前函數(shù)的id
5.互斥
創(chuàng)建互斥:
int pthread_mutex_init(pthread_mutex_t *restrict mutex,
const pthread_mutexattr_t *restrict attr);
參數(shù):
mutex輸出互斥id
attr 互斥屬性, 默認(rèn)NULL
note:函數(shù)成功返回0 否則返回錯誤碼
鎖住互斥:
int pthread_mutex_lock(pthread_mutex_t *mutex);
參數(shù):
mutex互斥id
note:如果指定的互斥id已經(jīng)被鎖住那么呼叫線程在互斥id完全解鎖前將一直處于掛起狀態(tài),否則將鎖住互斥體。
int pthread_mutex_trylock(pthread_mutex_t *mutex);
參數(shù):
mutex互斥id
note:如果指定的互斥id已經(jīng)被鎖住那么將直接返回一個錯誤,通過判斷此錯誤來進(jìn)行不同的處理。pthread_mutex_trylock和pthread_mutex_lock相似,不同的是pthread_mutex_trylock只有在互斥被鎖住的情況下才阻塞。
解鎖互斥:
int pthread_mutex_unlock(pthread_mutex_t *mutex);
參數(shù):
mutex互斥id
note:如果指定的互斥id已經(jīng)被鎖住那么對其解鎖
釋放互斥:
int pthread_mutex_destroy(pthread_mutex_t *mutex);
參數(shù):
mutex互斥id
note:釋放指定的mutex占用的資源。
函數(shù)pthread_mutex_init和pthread_mutex_destroy分別是互斥鎖的構(gòu)造函數(shù)和析構(gòu)函數(shù)。
二、多線程同步
1.互斥體
互斥量(mutex)相當(dāng)于一把鎖,可以保證以下三點:
◎原子性:如果一個線程鎖定一個互斥量,那么臨界區(qū)內(nèi)的操作要么全部完成,要么一個也不執(zhí)行。
◎惟一性:如果一個線程鎖定一個互斥量,那么在它解除鎖定之前,沒有其他線程可以鎖定這個互斥量。
◎非繁忙等待:如果一個線程已經(jīng)鎖定一個互斥量,第二個線程又試圖去鎖定這個互斥量,則第二個線程將被掛起(不占用任何CPU資源),直到第一個線程解除對這個互斥量的鎖定為止。
2.條件變量
條件變量是一種可以使線程(不消耗CPU)等待某些事件發(fā)生的機(jī)制。某些線程可能守候著一個條件變量,直到某個其他的線程給這個條件變量發(fā)送一個信號,這時這些線程中的一個線程就會蘇醒,處理這個事件。但條件變量不提供鎖定,所以它必須與一個互斥量同時使用,提供訪問這個環(huán)境變量時必要的鎖定。
3.信號量
Dijkstra提出了信號量的概念,信號量是一種特殊的變量,只可以取正整數(shù)值,對這個正整數(shù)只能采取兩種操作:P操作(代表等待,關(guān)操作)和V操作(代表信號,開操作)。
P/V操作定義如下(假設(shè)我們有一個信號量sem) :
P(sem):如果sem的值大于0,則sem減1;如果sem的值為0,則掛起該線程。
V(sem):如果有其它進(jìn)程因等待sem而掛起,則讓它恢復(fù)執(zhí)行;如果沒有線程等待sem而被掛起,則sem加上1。
信號集的創(chuàng)建與打開
int semget(key_t key,int nsems,int flag);
對信號量的操作
int semop(int semid,struct sembuf semoparray[],size_t nops);
對信號量的控制
int semctl(int semid,int semnum int cmd,union semun arg);
附:經(jīng)典的生產(chǎn)者-消費者問題(Producer-Costomer)是一個著名的同步問題。
相關(guān)文章
Linux系統(tǒng)下netstat命令詳細(xì)介紹
大家好,本篇文章主要講的是Linux系統(tǒng)下netstat命令詳細(xì)介紹,感興趣的同學(xué)趕快來看一看吧,對你有幫助的話記得收藏一下,方便下次瀏覽2021-12-12FOXHIS PMS前臺西軟服務(wù)器端和客戶端安裝教程
網(wǎng)上有很多關(guān)于西軟FOXHIS PMS的安裝教程,但大多沒有詳細(xì)描述并且內(nèi)容非常膚淺粗燥,本特意把自己的安裝過程寫成文檔并記下,后面的數(shù)據(jù)庫操作沒有提供圖片,但是我相信絕對能夠看得懂2017-01-01linux 網(wǎng)絡(luò)編程 socket選項的實現(xiàn)
這篇文章主要介紹了linux 網(wǎng)絡(luò)編程 socket選項的實現(xiàn),小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2019-06-06Linux (Ubuntu 18.04) 下安裝vim編輯器的方法
這篇文章主要介紹了Linux (Ubuntu 18.04) 下安裝vim編輯器的方法,非常不錯,具有一定的參考借鑒價值,需要的朋友可以參考下2019-07-07Linux系統(tǒng)擴(kuò)容根目錄磁盤空間的操作方法
這篇文章主要介紹了Linux系統(tǒng)擴(kuò)容根目錄磁盤空間的操作方法,非常不錯,具有一定的參考借鑒價值,需要的朋友可以參考下2018-07-07apache中通過mod_rewrite實現(xiàn)偽靜態(tài)頁面的方法
mod_rewrite是Apache的一個非常強(qiáng)大的功能,它可以實現(xiàn)偽靜態(tài)頁面。下面我詳細(xì)說說它的使用方法!對初學(xué)者很有用的哦!2011-08-08