Linux系统编程8:多线程编程

1. 概念

线程是比进程更小的能独立运行的基本单位,线程基本上不拥有系统资源,只拥有一点在运行中必不可少的资源(如线程ID,一组寄存器和栈),但是它可与同属一个进程的其他的线程共享进程所拥有的全部资源。

一个线程包含以下内容:

  1. 指向当前被执行指令的指令指针;
  2. 栈;
  3. 寄存器值的集合,定义了一部分描述正在执行线程的处理器状态的值;
  4. 私有的数据区
进程与线程静态

进程与线程动态
  • 参考
man 7 threads

2. 查看线程

  • 命令
No. 命令 含义
1 ps -T -p <pid> -T开启线程查看
2 top -H -p <pid> -H开启线程查看
  • 文件
No. 文件 含义
1 /proc/{PID}/task/ 线程默认的名字和进程名相同
2 /proc/{PID}/task/{tid}/comm 线程名

3. 操作

No. 操作 函数
1 线程标识 pthread_t pthread_self(void)
2 线程创建 int pthread_create(pthread_t * tidp, pthread_attr_t * attr, void *(*start_rtn)(void), void * arg)
3 子线程终止 void pthread_exit(void* retval)
4 线程合并 int pthread_join(pthread_t tid, void **retval)
5 线程分离 int pthread_detach(pthread_t tid)
6 发送信号 int pthread_kill(pthread_t tid, int sig)

3.1 线程标识

pthread_t pthread_self(void)
  • 返回值
    当前线程的线程ID

说明:

  1. 线程ID打印使用%lu
  2. pthread_self()不链接库pthread返回值为0.

linux上的线程实现就是在内核支持的基础上以POSIX thread的方式对外封装了接口。

3.2 线程创建

int pthread_create(pthread_t * tidp, pthread_attr_t * attr, void *(*start_rtn)(void), void * arg)
  • 参数
No. 参数 含义
1 tidp 线程ID指针
2 attr 线程属性
  • 示例
    两个线程并发执行
#include <stdio.h>
#include <unistd.h>
#include <pthread.h>
#include <sys/prctl.h>
 
int info(){
    printf("PID:%d,TID:%lu",getpid(),pthread_self());
    char name[16] = {0};
    prctl(PR_GET_NAME,name);
    printf("TNAME:%s\n",name);
}
 
void* method(void* arg){
    int i = 0;
    for(;i<100;i++){
        usleep(500000);
        info();
    }
}
 
int main(){
    info();
    pthread_t tid;
    pthread_create(&tid,NULL,method,NULL);
    printf("new tid:%lu\n",tid);
    int i = 0;
    for(;i<100;i++){
        sleep(1);
        info();
    }
    sleep(1);
}

模拟抢票

#include <stdio.h>
#include <unistd.h>
#include <pthread.h>
#include <sys/prctl.h>
 
long ticket = 1000000;
 
void* method(void* arg){
    while(ticket > 0){
        ticket--;
        printf("%lu get a ticket, leave %d\n",pthread_self(),ticket);
        //sleep(1);
    }
}
 
int main(){
    pthread_t tid;
    pthread_setconcurrency(5);
    pthread_create(&tid,NULL,method,NULL);
    pthread_create(&tid,NULL,method,NULL);
    pthread_create(&tid,NULL,method,NULL);
    pthread_create(&tid,NULL,method,NULL);
    pthread_create(&tid,NULL,method,NULL);
    pause();
}

3.3 子线程退出

子线程退出有两种方式

  1. 线程处理函数return
  2. 调用子线程终止
void pthread_exit(void* retval)
  • 参数
No. 参数 含义
1 retval 函数的返回指针,只要pthread_join中的第二个参数retval不是NULL,这个值将被传递给retval

用在线程回调函数中,返回线程数据

3.4 线程合并

使用valgrind试试上面的程序是否有内存泄漏?

int pthread_join(pthread_t tid, void **retval)
  • 参数
No. 参数 含义
1 tid 被等待的线程标识符
2 retval 一个用户定义的指针,它可以用来存储被等待线程的返回值
  • 返回值
No. 返回值 含义
1 0 成功
2 非0 错误码

可以由其他线程终止,回收资源

  • 示例
    无传参的情况
#include <stdio.h>
#include <unistd.h>
#include <pthread.h>
#include <sys/prctl.h>
 
int info(){
    printf("PID:%d,TID:%lu",getpid(),pthread_self());
    char name[16] = {0};
    prctl(PR_GET_NAME,name);
    printf("TNAME:%s\n",name);
}
 
void* method(void* arg){
    sleep(5);
    info();
}
 
int main(){
    info();
    pthread_t tid;
    pthread_create(&tid,NULL,method,NULL);
    printf("new tid:%lu\n",tid);
    //sleep(1);
    pthread_join(tid,NULL);
}

有传参的情况

#include <stdio.h>
#include <unistd.h>
#include <pthread.h>
#include <sys/prctl.h>
 
int info(){
    printf("PID:%d,TID:%lu",getpid(),pthread_self());
    char name[16] = {0};
    prctl(PR_GET_NAME,name);
    printf("TNAME:%s\n",name);
}
 
void* method(void* arg){
    sleep(5);
    info();
    printf("arg:%s",arg);
}
 
int main(){
    info();
    pthread_t tid;
    char test[]="hello thread";
    pthread_create(&tid,NULL,method,test);
    printf("new tid:%lu\n",tid);
    //sleep(1);
    pthread_join(tid,NULL);
}
  • 线程参数
    局部变量可以作为线程参数的情况
#include <stdio.h>
#include <unistd.h>
#include <pthread.h>
#include <sys/prctl.h>
 
int info(){
    printf("PID:%d,TID:%lu",getpid(),pthread_self());
    char name[16] = {0};
    prctl(PR_GET_NAME,name);
    printf("TNAME:%s\n",name);
}
 
void* method(void* arg){
    sleep(5);
    info();
    printf("arg:%s",arg);
    return "this is return value";
}
 
int main(){
    info();
    pthread_t tid;
    char test[]="hello thread";
    pthread_create(&tid,NULL,method,test);
    printf("new tid:%lu\n",tid);
    //sleep(1);
    void* res = NULL;
    pthread_join(tid,&res);
    printf("res:%s\n",res);
}

局部变量不能作为线程参数的情况

#include <stdio.h>
#include <unistd.h>
#include <pthread.h>
#include <sys/prctl.h>
 
int info(){
    printf("PID:%d,TID:%lu",getpid(),pthread_self());
    char name[16] = {0};
    prctl(PR_GET_NAME,name);
    printf("TNAME:%s\n",name);
}
 
void* method(void* arg){
    sleep(5);
    info();
    printf("arg:%s",arg);
}
 
pthread_t create_thread(){
    pthread_t tid;
    char test[] = "Hello thread";
    pthread_create(&tid,NULL,method,(void*)test);
    return tid;
}
 
int main(){
    info();
    pthread_t tid;
    //char test[]="hello thread";
    //pthread_create(&tid,NULL,method,test);
    tid = create_thread();
    printf("new tid:%lu\n",tid);
    //sleep(1);
    pthread_join(tid,NULL);
}

只读变量可以作为线程参数的情况

#include <stdio.h>
#include <unistd.h>
#include <pthread.h>
#include <sys/prctl.h>
 
int info(){
    printf("PID:%d,TID:%lu",getpid(),pthread_self());
    char name[16] = {0};
    prctl(PR_GET_NAME,name);
    printf("TNAME:%s\n",name);
}
 
void* method(void* arg){
    sleep(5);
    info();
    printf("arg:%s",arg);
}
 
pthread_t create_thread(){
    pthread_t tid;
    const char* test = "Hello thread";
    pthread_create(&tid,NULL,method,(void*)test);
    return tid;
}
 
int main(){
    info();
    pthread_t tid;
    //char test[]="hello thread";
    //pthread_create(&tid,NULL,method,test);
    tid = create_thread();
    printf("new tid:%lu\n",tid);
    //sleep(1);
    pthread_join(tid,NULL);
}

堆变量可以作为线程参数的情况

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <pthread.h>
#include <sys/prctl.h>
 
int info(){
    printf("PID:%d,TID:%lu",getpid(),pthread_self());
    char name[16] = {0};
    prctl(PR_GET_NAME,name);
    printf("TNAME:%s\n",name);
}
 
void* method(void* arg){
    sleep(5);
    info();
    printf("arg:%s",arg);
    free(arg);
}
 
pthread_t create_thread(){
    pthread_t tid;
    char* test = malloc(BUFSIZ);
    strcpy(test,"Hello thread");
    pthread_create(&tid,NULL,method,(void*)test);
    return tid;
}
 
int main(){
    info();
    pthread_t tid;
    //char test[]="hello thread";
    //pthread_create(&tid,NULL,method,test);
    tid = create_thread();
    printf("new tid:%lu\n",tid);
    //sleep(1);
    pthread_join(tid,NULL);
}

静态变量可以作为线程参数的情况

#include <stdio.h>
#include <unistd.h>
#include <pthread.h>
#include <sys/prctl.h>
 
int info(){
    printf("PID:%d,TID:%lu",getpid(),pthread_self());
    char name[16] = {0};
    prctl(PR_GET_NAME,name);
    printf("TNAME:%s\n",name);
}
 
void* method(void* arg){
    sleep(5);
    info();
    printf("arg:%s",arg);
}
 
pthread_t create_thread(){
    pthread_t tid;
        static char test[] = "Hello thread";
    pthread_create(&tid,NULL,method,(void*)test);
    return tid;
}
 
int main(){
    info();
    pthread_t tid;
    //char test[]="hello thread";
    //pthread_create(&tid,NULL,method,test);
    tid = create_thread();
    printf("new tid:%lu\n",tid);
    //sleep(1);
    pthread_join(tid,NULL);
}

3.5 线程分离

int pthread_detach(pthread_t tid)
  • 参数
No. 参数 含义
1 tid 要释放线程的标识符ID
  • 返回值
No. 返回值 含义
1 0 成功
2 非0 错误码

不能被其他线程终止,存储资源在它终止时由系统自动回收释放
线程分离后不能使用join。

使用valgrind试试pthread_detach()的程序是否有内存泄漏?

4. 进程线程比较

4.1 接口对比

No. Process Primitive Thread Primitive Description
1 getpid() pthread_self() 获得控制流的id
2 fork() pthread_create() 创建新的控制流
3 exit() pthread_exit() 退出已有的控制流
4 waitpid() pthread_join() 等待控制流并获得结束代码

4.2 特性对比

No. 特性 进程 线程
1 粒度 系统资源分配和调度的基本单位 CPU调度和分派的基本单位
2 资源 有独立的地址空间 共享进程的地址空间
3 效率 上下文切换要较慢 上下文切换较快
4 稳定性 子进程崩溃,不影响父进程与其他子进程 任何一个线程崩溃,整个程序崩溃

5 线程通信

全局变量、动态分配内存和比线程生存周期长的局部变量都可以用于线程通信。

线程通信比进程简单方便。

问题:尽然有了进程,为什么要有线程?

6. C++11 thread

使用pthread实现简单的thread类。

#ifndef __THREAD_H
#define __THREAD_H
#include <pthread.h>

namespace miniSTL{
    class thread{
        typedef pthread_t id;
        typedef void* (*func_t)(void*);
        typedef void (*funcv_t)();
        id _id; // 线程TID
    public:
         thread():_id(0){}
         thread(func_t func){
            pthread_create(&_id,NULL,func,NULL);
         }
         thread(funcv_t func){
            pthread_create(&_id,NULL,reinterpret_cast<func_t>(func),NULL);
         }
         id get_id()const{
            return _id;
         }
         void join(){
            pthread_join(_id,NULL);
         }
         void detach(){
            pthread_detach(_id);
         }
    };

};
#endif // __THREAD_H
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,732评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,496评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,264评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,807评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,806评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,675评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,029评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,683评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 41,704评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,666评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,773评论 1 332
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,413评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,016评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,978评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,204评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,083评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,503评论 2 343

推荐阅读更多精彩内容