Linux系统编程—线程创建和终止

线程的资源

在同一个进程中,大部分资源都是所有线程所共享的,比如:
可执行程序的代码
程序的全局内存
堆内存

文件描述符

但是线程还有一些独有的资源,包含:
线程 ID
线程自己的一套寄存器值
线程运行栈
调度优先级和策略
信号屏蔽字
errno 变量
线程私有数据

进程与线程的区别根本区别:

进程是操作系统资源分配的基本单位
线程是任务调度和执行的基本单位

在开销方面:
每个进程都有独立的代码和数据空间(程序上下文),程序之间的切换会有较大的开销;
线程可以看做轻量级的进程,同一类线程共享代码和数据空间,每个线程都有自己独立的运行栈和程序计数器(PC),线程之间切换的开销小。

所处环境:
在操作系统中能同时运行多个进程(程序);
而在同一个进程(程序)中有多个线程同时执行(通过CPU调度,在每个时间片中只有一个线程执行)

内存分配方面:
系统在运行的时候会为每个进程分配不同的内存空间;
而对线程而言,除了CPU外,系统不会为线程分配内存(线程所使用的资源来自其所属进程的资源),线程组之间只能共享资源。

包含关系:没有线程的进程可以看做是单线程的,如果一个进程内有多个线程,则执行过程不是一条线的,而是多条线(线程)共同完成的;
线程是进程的一部分,所以线程也被称为轻权进程或者轻量级进程。

pthread 线程库

pthread 是 POSIX.1 中定义的一套线程函数接口,所以这种线程也被称为 POSIX 线程。在我们后面的学习中,都会使用 pthread 线程库来操作线程,包括线程的创建、终止、互斥与同步等等。
要看你的的系统是否支持 POSIX 线程,可以判断宏 _POSIX_THREADS 是否被定义。

线程创建

函数原型:

typedef void *(*start_routine) (void *);
int pthread_create(pthread_t *thread, const pthread_attr_t *attr, start_routine th_fn, void *arg);

参数1:
pthread_t 类型是线程 id 的类型,它可能是个整数(大多数是这样),也可能是结构体,这取决于具体实现。
thread 参数是一个传出参数,它主要用来接收被创建的线程的 id 号。在线程执行过程中调用函数pthread_self函数,可以获取到自己的线程 id 号。

参数2:
pthread_attr_t 类型是用来保存线程属性的结构体
参数 attr 可以传 NULL,表示创建具有默认属性的线程。有关线程的属性结构体,后文会作具体讲解。

参数3:
start_routine 是用来声明函数型如 void* th_fn(void* arg) 这样的函数的,它是线程过程函数。线程被创建后,一旦被调度,就会进入到函数 th_fn 中执行。

参数4:
最后一个参数 arg,是线程过程函数 th_fn 的参数。

返回值:
pthread_create 函数的返回值是一个错误码,返回 0 表示成功。和其它 POSIX 函数不同的是,pthread_create 函数执行错误并不会设置 errno 变量。

#include <unistd.h>
#include <pthread.h>
#include <stdio.h>

// 线程入口函数
void* th_fn(void *arg) {
    char* name = (char*)arg;
    int i;

    for (i = 1; i <= 10; ++i) {
        printf("%s : %d\n", name, i);

        // 判断是 张三 还是 李四
        if (strcmp("张三", name) == 0){
            usleep(1000*10); // 10 ms
        }
        else{
            usleep(1000*5); // 5ms
        }
    }

    return NULL;
}


int main() {
    int err;
    pthread_t p1, p2;
    char *name1 = "张三";
    char *name2 = "李四";

    // 创建张三线程。
    err = pthread_create(&p1, NULL, th_fn, (void*)name1);
    if (err != 0) {
        perror("pthread_create1 error\n");
        return -1;
    }
    // 创建李四线程
    err = pthread_create(&p2, NULL, th_fn, (void*)name2);
    if (err != 0) {
        perror("pthread_create2 error\n");
        return -1;
    }

    //如果没有此行,线程效果无法显示
    sleep(3);

    // 打印线程 id 号
    printf("张三 id: %lx\n", p1);
    printf("李四 id: %lx\n", p2);

    return 0;
}

编程和运行

// 注意需要链接 pthread 库
gcc thread.c -o thread -lpthread
./hellothread

线程终止

线程终止方式线程的终止包括两大类,主动终止和被动终止。

主动终止

线程过程函数执行 return 正常返回,返回值是线程的退出码
线种过程函数执行 pthread_exit 函数退出,其参数是线程退出码

被动终止

在其它线程中调用 pthread_cancel 函数
任意线程调用 exit、_Exit 或者 _exit 导致整个进程终止

pthread_exit 函数

pthread_exit 函数原型

void pthread_exit(void *rval_ptr);

该函数的参数是线程的退出码,它是无类型的指针。通过函数 pthread_join 可以获取到这个指针的值,它可以回收线程所使用的资源。

int pthread_join(pthread_t tid, void **rval_ptr);

调用pthread_join 会阻塞当前线程,直到指定的线程 tid 返回(return)、执行 pthread_exit 或者被其它线程取消(pthread_cancel)。rval_ptr 是一个输出参数,它用来接收线程退出码(是一个 void* 类型)。

pthread_cancel 函数

int pthread_cancel(pthread_t tid);

默认情况下,该函数的效果相当于在线程 tid 中调用 pthread_exit((void*)PTHREAD_CANCEL) 函数。

线程主动退出实例:

#include <unistd.h>
#include <pthread.h>
#include <stdio.h>
#include <string.h>


void* th_fn1(void* arg) {
    puts("thread 1 returning");
    return (void*)10;
}

void* th_fn2(void* arg) {
    puts("thread 2 exiting");
    pthread_exit((void*)20);
    puts("thread 2 exited");
}

int main() {
    pthread_t tid1, tid2;
    int err;
    void* ret;

    err = pthread_create(&tid1, NULL, th_fn1, NULL);
    if (err != 0) {
        perror("pthread_create1 error\n");
        return -1;
    }

    err = pthread_create(&tid2, NULL, th_fn2, NULL);
    if (err != 0) {
        perror("pthread_create2 error\n");
        return -1;
    }

    sleep(2);

    // 主线程阻塞直到线程 tid1 退出
    err = pthread_join(tid1, &ret);
    if (err != 0) {
        perror("pthread_join1 error\n");
        return -1;
    }
    printf("thread 1 exit code %d\n", (int)ret);

    // 主线程阻塞直到线程 tid2 退出
    err = pthread_join(tid2, &ret);
    if (err != 0) {
        perror("pthread_join2 error\n");
        return -1;
    }
    printf("thread 2 exit code %d\n", (int)ret);

    return 0;
}

image.png

被动终止实例:

#include <unistd.h>
#include <pthread.h>
#include <stdio.h>
#include <string.h>


void* th_fn1(void* arg) {
    while(1) {
        puts("thread 1 running");
        sleep(1);
    }
    return (void*)10;
}

void* th_fn2(void* arg) {
    while(1) {
        puts("thread 2 running");
        sleep(1);
    }
    pthread_exit((void*)20);
}

int main() {
    pthread_t tid1, tid2;
    int err;
    void* ret;
    
    err = pthread_create(&tid1, NULL, th_fn1, NULL);
    if (err != 0) {
        perror("pthread_create1 error\n");
        return -1;
    }
    
    err = pthread_create(&tid2, NULL, th_fn2, NULL);
    if (err != 0) {
        perror("pthread_create2 error\n");
        return -1;
    }
    
    sleep(5);
    
    // 通知 tid1 和 tid2 退出。
    pthread_cancel(tid1);
    pthread_cancel(tid2);
    
    err = pthread_join(tid1, &ret);
    
    // 线程退出码都是 PTHREAD_CANCELED
    printf("PTHREAD_CANCELED = %d\n", (int)PTHREAD_CANCELED);
    if (err != 0) {
        perror("pthread_join1 error\n");
        return -1;
    }
    printf("thread 1 exit code %d\n", (int)ret);
    
    err = pthread_join(tid2, &ret);
    if (err != 0) {
        perror("pthread_join2 error\n");
        return -1;
    }
    printf("thread 2 exit code %d\n", (int)ret);
    
    return 0;
}
image.png
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 217,084评论 6 503
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,623评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 163,450评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,322评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,370评论 6 390
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,274评论 1 300
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,126评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,980评论 0 275
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,414评论 1 313
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,599评论 3 334
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,773评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,470评论 5 344
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,080评论 3 327
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,713评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,852评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,865评论 2 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,689评论 2 354

推荐阅读更多精彩内容