亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > C++ > 正文

linux下c語言的多線程編程

2020-05-23 13:46:13
字體:
來源:轉載
供稿:網友

我們在寫linux的服務的時候,經常會用到linux的多線程技術以提高程序性能 

多線程的一些小知識:

一個應用程序可以啟動若干個線程。

線程(Lightweight Process,LWP),是程序執行的最小單元。

一般一個最簡單的程序最少會有一個線程,就是程序本身,也就是主函數(單線程的進程可以簡單的認為只有一個線程的進程)

 一個線程阻塞并不會影響到另外一個線程。

多線程的進程可以盡可能的利用系統CPU資源。

1創建線程

先上一段在一個進程中創建一個線程的簡單的代碼,然后慢慢深入。

#include<pthread.h>#include<stdio.h>#include<stdlib.h>#include<string.h>#include<errno.h>void * func(void * arg){ printf("func run.../n"); return NULL;}int main(){ pthread_t t1; int err = pthread_create(&t1,NULL,func,NULL); if(err!=0) {  printf("thread_create Failed:%s/n",strerror(errno)); }else{  printf("thread_create success/n"); } sleep(1); return EXIT_SUCCESS;}int pthread_create(pthread_t *thread,const pthread_attr_t *attr, void *(*start_routine)(void*), void *arg);

在main函數里面我們調用上面的函數進行創建一個線程。

函數參數:

  第一個參數:pthread_t代表創建線程的唯一標識,是一個結構體,需要我們創建好后,將這個結構體的指針傳遞過去。

  第二個參數:pthread_attr_t,代表創建這個線程的一些配置,比如分配棧的大小等等。。一般我們可以填NULL,代表默認的創建線程的配置

  第三個參數:代表一個函數的地址,創建線程時,會調用這個函數,函數的返回值是void*,函數的參數也是void*,一般格式就像void * func(void * arg){}

  第四個參數:代表調用第三個函數傳遞的參數

函數返回值:

  函數成功返回0,如果不等于0則代表函數調用失敗,此時通過strerror(errno)可以打印出具體的錯誤。

  注意:每個線程都擁有一份errno副本,不同的線程擁有不同的errno

最后通過gcc編譯

gcc 1createthread.c -c -o 1createthread.ogcc 1createthread.o -o thr1 -lpthread

編譯的時候需要加上-lpthread 用來鏈接libpthread.so動態庫,不然會提示找不到function

函數調用返回結果

linux下多線程編程,linux,c語言,多線程編程

問題:為什么調用sleep函數

答:可能新創建的線程還沒運行到打印的方法主線程就結束了,而主線程結束,所有線程都會結束了。

2線程掛起

有時候我們在一個線程中創建了另外一個線程,主線程要等到創建的線程返回了,獲取該線程的返回值后主線程才退出。這個時候就需要用到線程掛起。

int pthread_join(pthread_t th, void **thr_return);。

pthread_join函數用于掛起當前線程,直至th指定的線程終止為止。

#include<pthread.h>#include<stdio.h>#include<stdlib.h>#include<string.h>#include<errno.h>void * func(void * arg){ int i=0; for(;i<5;i++) {  printf("func run%d/n",i);  sleep(1); } int * p = (int *)malloc(sizeof(int)); *p=11; return p;}int main(){ pthread_t t1,t2; int err = pthread_create(&t1,NULL,func,NULL); if(err!=0) {  printf("thread_create Failed:%s/n",strerror(errno)); }else{  printf("thread_create success/n"); } void *p=NULL; pthread_join(t1,&p); printf("線程退出:code=%d/n",*(int*)p); return EXIT_SUCCESS;}

函數執行結果

linux下多線程編程,linux,c語言,多線程編程

我們主函數一直在等待創建的線程執行完,并且得到了線程執行結束的返回值

3線程終止

進程終止時exit()函數,那么線程終止是什么呢?

線程終止的三種情況:

線程只是從啟動函數中返回,返回值是線程的退出碼。

線程可以被同一進程中的其他線程取消。

線程調用pthread_exit。

#include<pthread.h>#include<stdio.h>#include<stdlib.h>#include<string.h>#include<errno.h>void * func(void * arg){ int i=0; while(1) {  if(i==10)  {   int * p = (int *)malloc(sizeof(int));   *p=11;   pthread_exit(p);  }  printf("fun run %d/n",i++);  sleep(1); } return NULL;}int main(){ pthread_t t1,t2; int err = pthread_create(&t1,NULL,func,NULL); if(err!=0) {  printf("thread_create Failed:%s/n",strerror(errno)); }else{  printf("thread_create success/n"); } void *p=NULL; pthread_join(t1,&p); printf("線程退出:code=%d",*(int*)p); return EXIT_SUCCESS;}void pthread_exit(void *arg);

pthread_exit函數的參數就跟正常線程結束return的使用時一樣的,都會被等待它結束的主線程獲取到。

函數運行結果:

linux下多線程編程,linux,c語言,多線程編程

4線程分離

int pthread_detach(pthread_t th);

pthread_detach函數使線程處于被分離狀態。

如果不等待一個線程,同時對線程的返回值不感興趣,可以設置這個線程為被分離狀態,讓系統在線程退出的時候自動回收它所占用的資源。

一個線程不能自己調用pthread_detach改變自己為被分離狀態,只能由其他線程調用pthread_detach。

5線程取消

int pthread_cancel(pthread_t th);

pthread_cancel函數允許一個線程取消th指定的另一個線程。

函數成功,返回0,否則返回非0。

#include<pthread.h>#include<stdio.h>#include<stdlib.h>#include<string.h>#include<errno.h>void * func1(void * arg){ while(1) {  printf("fun run.../n");  sleep(1); } return NULL;}int main(){ pthread_t t1; if(pthread_create(&t1,NULL,func1,NULL)!=0) {  printf("thread_create Failed:%s/n",strerror(errno));  return -1; } sleep(5); pthread_cancel(t1); pthread_join(t1,NULL); return EXIT_SUCCESS;}

 函數執行結果:

linux下多線程編程,linux,c語言,多線程編程

上面我們說過創建一個線程函數pthread_create的第二個參數,用來決定創建線程的一些初始化狀態,這里我們 舉個例子,改線程一創建就是分離狀態的線程(

上面介紹了pthread_detach函數的概念,可以通過pthread_attr_t在創建線程的時候就指定線程屬性為detach,而不用創建以后再去修改線程屬性。

先上一段代碼:

#include<pthread.h>#include<stdio.h>#include<stdlib.h>#include<string.h>#include<errno.h>void * func(void * arg){ int i=0; for(;i<5;i++) {  printf("func run%d/n",i);  sleep(1); } int * p = (int *)malloc(sizeof(int)); *p=11; return p;}int main(){ pthread_t t1; pthread_attr_t attr;//申明一個attr的結構體 pthread_attr_init(&attr);//初始化結構體 pthread_attr_setdetachstate(&attr, PTHREAD_CREATE_DETACHED);//設置線程為分離線程 int err = pthread_create(&t1,&attr,func,NULL); if(err!=0) {  printf("thread_create Failed:%s/n",strerror(errno)); }else{  printf("thread_create success/n"); } pthread_attr_destroy(&attr); pthread_join(t1,NULL); printf("主線程退出/n"); return EXIT_SUCCESS;}

pthread_attr_t就是我們要傳入的參數的結構體,一般申明的步驟有

1,申明一個pthread_attr_t對象

2,函數pthread_attr_init初始化attr結構。

3,設置線程的一些屬性,比如pthread_attr_setdetachstate函數就是設置該線程創建的時候為正常狀態還是分離狀態。

4,函數pthread_attr_destroy釋放attr內存空間

pthread_attr_setdetachstate把線程屬性設置為下面兩個合法值之一:

 

說明

PTHREAD_CREATE_DETACHED

設置線程為分離狀態

PTHREAD_CREATE_JOINABLE

設置線程為正常狀態

 

上面函數運行結果:

linux下多線程編程,linux,c語言,多線程編程

因為線程是個分離狀態的,所以pthread_join掛起會失效,主線程很快運行結束,程序也就結束了,創建的線程還沒來得及運行

線程同步

有時候我們多個線程處理訂單扣減庫存會遇到這樣的問題,兩個線程同時進入一段代碼先查詢庫存,兩個都查出來為還剩一件庫存,第一個線程用掉這個庫存后,將庫存變為0,但是第二個線程剛才也查出來為1了,所以他還認為有庫存,

這個時候操作就會引發我們想不到的意外,庫存變為負數了??!所以這個時候就需要使用線程的同步??!

先上一段代碼看看效果:

#include<pthread.h>#include<stdio.h>#include<pthread.h>#include<stdio.h>#include<stdlib.h>#include<string.h>#include<errno.h>void * func(void * arg){ int threadno =*(int*)arg; int i=0; for(;i<10;i++) {  printf("%d thread%d /n",threadno,i);  sleep(1); } return NULL;}int main(){ pthread_t t1,t2; int i1=1,i2=2; pthread_create(&t1,NULL,func,&i1); pthread_create(&t2,NULL,func,&i2); pthread_join(t1,NULL); pthread_join(t2,NULL); printf("主線程退出/n"); return EXIT_SUCCESS;}

函數運行結果:

linux下多線程編程,linux,c語言,多線程編程

可以看到兩個線程是沒有規律的爭相處理的,如果這段代碼是扣減庫存就完蛋啦!,所以我們要對這段代碼進行加鎖,同一時刻只能有一個線程進入操作!

先上代碼:

#include<pthread.h>#include<stdio.h>#include<stdlib.h>#include<string.h>#include<errno.h>pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;void * func(void * arg){ pthread_mutex_lock(&mutex);//對mutex加鎖,其他線程進入后將會掛起,知道這個鎖被解鎖 int threadno =*(int*)arg; int i=0; for(;i<10;i++) {  printf("%d thread%d /n",threadno,i);  sleep(1); } pthread_mutex_unlock(&mutex); return NULL;}int main(){ pthread_t t1,t2; int i1=1,i2=2; pthread_create(&t1,NULL,func,&i1); pthread_create(&t2,NULL,func,&i2); pthread_join(t1,NULL); pthread_join(t2,NULL); printf("主線程退出/n"); return EXIT_SUCCESS;}

函數運行結果:

linux下多線程編程,linux,c語言,多線程編程

可以看到第二個線程先進入后一直運行結束,對mutex解鎖后,第一個線程才能進方法里面運行!否則會掛起,一直等到鎖被解鎖!

PTHREAD_MUTEX_INITIALIZER是初始化一個快速鎖的宏定義。

pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;

加鎖解鎖函數:

int pthread_mutex_lock(pthread_mutex_t *mutex);int pthread_mutex_unlock(pthread_mutex_t *mutex);

總結

以上所述是小編給大家介紹的linux下c語言的多線程編程,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復大家的。在此也非常感謝大家對VEVB武林網網站的支持!


發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
国产精品久久久久77777| 欧洲午夜精品久久久| 中文字幕日韩精品在线观看| 黄网动漫久久久| 精品亚洲一区二区三区四区五区| 国产欧美日韩免费看aⅴ视频| 不卡av在线网站| 国产成人精品网站| 国产91在线高潮白浆在线观看| 欧美极度另类性三渗透| 亚洲欧美日韩一区二区在线| 2020久久国产精品| 日韩欧美在线观看视频| 国产一区深夜福利| 久久成人精品一区二区三区| 国产成人jvid在线播放| 欧美性极品少妇精品网站| 98精品国产高清在线xxxx天堂| 性欧美xxxx交| 国产日韩精品一区二区| 91经典在线视频| 亚洲免费人成在线视频观看| 成人午夜两性视频| 欧美日韩成人黄色| 国产成人综合精品| 狠狠久久亚洲欧美专区| 中日韩美女免费视频网址在线观看| 亚洲人成在线电影| 91色精品视频在线| 8x海外华人永久免费日韩内陆视频| 亚洲精品成人网| 精品综合久久久久久97| 欧美成年人视频网站欧美| 欧美成人精品激情在线观看| 亚洲自拍小视频免费观看| 亚洲在线免费看| 国产精品99久久久久久人| 成人午夜在线视频一区| 国产成人精品免高潮在线观看| 成人a级免费视频| 国产成人精品综合久久久| 午夜精品在线观看| 亚洲直播在线一区| 日韩电影在线观看永久视频免费网站| 欧美成人在线免费| 国产日韩欧美在线看| 亚洲福利精品在线| 国产69精品久久久久9999| 欧美极品xxxx| 亚洲国产日韩欧美在线图片| 91精品国产91久久久久久最新| 欧美激情一二三| 久久久免费观看视频| 高清欧美性猛交xxxx| 日本亚洲欧洲色| 亚洲xxxx在线| 欧美精品xxx| 夜夜嗨av色综合久久久综合网| 亚洲午夜女主播在线直播| 精品一区二区三区四区| 91精品久久久久久久| 国产精品自在线| 亚洲激情免费观看| 91九色综合久久| 亚洲成人久久久| 国产丝袜精品第一页| 国产97色在线|日韩| 最近中文字幕2019免费| 亚洲va码欧洲m码| 国产视频久久久久| 欧日韩在线观看| 日韩在线观看免费高清| 色悠悠久久88| 91av在线免费观看| 91麻豆国产语对白在线观看| 一本色道久久综合狠狠躁篇怎么玩| 91人成网站www| 日韩av在线免费| 黄色成人av在线| 亚洲国产精品一区二区久| 成人国产精品久久久久久亚洲| 日韩精品在线影院| 亚洲视频视频在线| 日本成人精品在线| 国产区精品视频| 久久视频国产精品免费视频在线| 欧美成年人网站| 91在线视频成人| 欧美一级视频在线观看| 成人中心免费视频| 国产日韩视频在线观看| 青青草成人在线| 欧美成人性生活| 一本色道久久88精品综合| 日韩欧美国产高清91| 欧美大尺度在线观看| 免费91在线视频| 色老头一区二区三区| 亚洲第一视频在线观看| 欧美体内谢she精2性欧美| 精品国产一区二区三区在线观看| 青草成人免费视频| 久久人体大胆视频| 日韩av电影免费观看高清| 麻豆国产va免费精品高清在线| 国产精品日本精品| 精品视频中文字幕| 国产视频精品va久久久久久| 在线播放精品一区二区三区| 欧美性猛交xxxx免费看漫画| 成人国产精品免费视频| 国产午夜精品一区理论片飘花| 91在线中文字幕| 欧美成人四级hd版| 国产精品狼人色视频一区| 精品人伦一区二区三区蜜桃网站| 亚洲精品国产精品国产自| 九色精品免费永久在线| 一区二区三区久久精品| 国产精品中文字幕久久久| 欧美日韩高清在线观看| 欧美成年人网站| 亚洲最大的成人网| 久久网福利资源网站| 亚洲精品电影在线观看| 91夜夜未满十八勿入爽爽影院| 亚洲欧洲日产国产网站| 91福利视频网| 综合国产在线视频| 国产91精品不卡视频| 国产亚洲美女久久| 亚洲欧美国产日韩天堂区| 国产成人精品在线视频| 国产精品va在线播放我和闺蜜| 国产综合在线看| 九九热r在线视频精品| 欧美午夜丰满在线18影院| 91亚洲精品久久久久久久久久久久| 欧美xxxx综合视频| 欧美小视频在线观看| 欧美一区深夜视频| 日日摸夜夜添一区| 国产精品久久久久久av下载红粉| 国产这里只有精品| 九九综合九九综合| 精品成人国产在线观看男人呻吟| 色偷偷888欧美精品久久久| 九色精品免费永久在线| 亚洲性猛交xxxxwww| 国产精品久久久久久久9999| 亚洲一区二区三区乱码aⅴ| 在线电影av不卡网址| 亚洲欧美中文日韩v在线观看| 成人在线激情视频| 亚洲一区二区三区乱码aⅴ蜜桃女| 日韩麻豆第一页| 国产精品亚洲视频在线观看| 91精品久久久久久久久青青| 欧美成在线视频| 国产一区二区久久精品| 亚洲在线视频福利| 日韩av在线一区二区| 国产91露脸中文字幕在线| 欧美日韩亚洲91|