Linux线程之线程的创建、属性、回收、退出、取消方式

2025-07-11 18:50

本文主要是介绍Linux线程之线程的创建、属性、回收、退出、取消方式,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

《Linux线程之线程的创建、属性、回收、退出、取消方式》文章总结了线程管理核心知识:线程号唯一、创建方式、属性设置(如分离状态与栈大小)、回收机制(join/detach)、退出方法(返回/pthr...

1. 线程号

进程号在系统中唯一,但线程号只在其所属进程环境中有效。

(1)pthread_self函数

#include<pthread.h>

pthread_t pthread_self(void);
/*
功能:
    获取线程号
返回值:
    调用此函数线程的ID
*/

pthread_self示例:

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

int main(int argc, const char* argv[]) {
    pthread_t tid = 0;
    tid = pthread_self();
    printf("当前线程id:%lu.\n", tid);
    return 0;
}

编译时需要加上-pthread链接到pthread库

运行结果:

Linux线程之线程的创建、属性、回收、退出、取消方式

(2)pthread_equal函数

int pthraed_equal(pthread_t t1, pthread_t t2);
/*
功能:
    判断线程号t1、t2是否相等。
返回值:
 php   相等:非0
    不等:0
*/

pthread_equal示例:

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

int main(int argc, const char* argv[]) {
    pthread_t tid = 0;
    tid = pthread_self();

    if (pthread_equal(tid, pthread_self())) {
        printf("线程id相等.\n");
    } else {
        printf("线程id不等.\n");
    }
    return 0;
}

运行结果:

Linux线程之线程的创建、属性、回收、退出、取消方式

2. 线程的创建

pthread_create函数

#include<pthread.h>

int pthread_create(pthread_t* thread, const pthread_attr_t* attr, 
                    void* (*start_coutine)(void*), void* arg);
/*
功能:
    创建一个线程。
参数:
    thread:线程id地址,为传出参数;
    attr:线程属性结构体,通常设置为NULL;
    start_routine:线程函数入口地址
    arg:传给线程函数的参数;
返回值:
    成功:0
    失败:非0,未设置errno,不可使用perror。
*/

pthread_create示例:

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

void* threadFunc(void* arg) { // 线程调度函数
    int var = (int)(long)(arg);
    printf("被创建线程id:%lu,传过来的参数:%d\n", pthread_self(), var);
    return NULL;
}

int main(int argc, const char* argv[]) {

    pthread_tEpMiaGnMt tid;
    int ret = -1;

    // 初始化tid。因为不是所有系统中的pthread_t都是unsigned int, 因此最好使用memset初始化。
    memset(&tid, 0, sizeof(tid));

    // 创建线程
    ret = pthread_create(&tid, NULL, threadFunc, (void*)0x3);
    if (0 != ret) {
        printf("线程创建失败!\n");
        return 1;
    }
    printf("按下任意键继续...\n");
    getchar();
    printf("主线程id:%lu\n", pthread_self());

    return 0;
}

运行结果:

Linux线程之线程的创建、属性、回收、退出、取消方式

3. 线程属性

typedef struct {
    int etachstate;   // 线程的分离状态
    int schedpolicy;  // 线程的调度策略
    struct sched_param schedparam; // 线程的调度参数
    int inheritsched;  // 线程的继承性
    int scope;         // 线程的作用域
    size_t guardsize;  // 线程栈末尾的警戒缓冲区大小
    int stackaddr_set; // 线程栈的设置
    void* stackaddr;   // 线程栈的位置
    size_t stacksize;  // 线程栈的大小
} pthread_attr_t;
/*
功能:
    线程属性结构体;
主要成员:
    etachstate:线程的分离状态
    guardsize:线程栈末尾的警戒缓冲区大小
    stackaddr:线程栈的位置
    stacksize:线程栈的大小

注意:
    线程属性值不能直接设置,需使用相关函数进行操作。
    如pthread_create之前用pthread_attr_init初始化,
    之后用pthread_attr_destory释放资源。
*/

(1)线程属性的初始化和销毁

#include<pthread.h>

int pthread_attr_init(pthread_attr_t* attr);
/*
功能:
    初始化线程属性attr。
参数:
    attr:待初始化的线程属性结构体。
返回值:
    成功:0
    失败:错误码
*/

int pthread_attr_destory(pthread_attr_t* attr);/*
功能:
    销毁线程属性attr。
参数:
    attr:线程属性结构体。
返回值:
    成功:0
    失败:非0错误码
*/

(2)线程分离状态属性设置

  • 线程分离状态:无需其他线程阻塞等待线程结束以回收已结束线程的资源,而是让内核回收已结束线程的资源。
  • 线程非分离状态:某个线程(如主线程)阻塞等待子线程结束以回收其资源。
#include<pthread.h>

int pthread_attr_setdetachstate(pthread_attr_t* attr, int detachstate);
/*
功能:
    设置线程属性为分离。
参数:
    attr:已初始化的线程属性结构体;
    detachstate:是否分离:
        PTHREAD_CREATE_DETACHED:分离
        PTHREAD_CREATE_JOINABLE:非分离
返回值:
    成功:0
    失败:非0
*/

int pthread_attr_getdetachstate(const pthread_attr_t* attr, int* detachstate);
/*
功能:
    获取线程是否分离状态
参数:
    attr:线程属性结构体。
        detachstate:是否分离:
        PTHREAD_CREATE_DETACHED:分离
        PTHREAD_CREATE_JOINABLE:非分离
返回值:
    成功:0
    失败:非0错误码
*/

线程设置分离属性示例

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

void* func(void* arg) {
    printf("子线程开始...\n");
    sleep(2);
    printf("子线程结束...\n");
    pthread_exit(NULL);
}

int main(int argc, const char* argv[]) {

    int ret = -1;
    pthread_t tid = -1;
    pthread_attr_t attr;

    // 初始化线程属性
    ret = pthread_attr_init(&attr);
    if (0 != ret) {
        printf("线程属性初始化失败。\n");
        return 1;
    }

    // 设置线程属性为分离状态
    ret = pthread_attr_setdetachstate(&attr, PTHREAD_CREATE_DETACHED);
    if (0 != ret) {
        printf("线程分离属性设置失败。\n");
        return 1;
    }

    // 创建线程
    ret = pthread_create(&tid, &attr, func, NULL);
http://www.chinasem.cn    if (0 != ret) {
        printf("线程创建失败.\n");
        return 1;
    }

    sleep(3);

    // join看看是否为分离状态
    ret = pthread_join(tid, NULL);
    if (0 != ret) {
        printf("线程为分离状态,无需join.\n");
    } else {
        printf("线程为非分离状态,已被join.\n");
    }

    // 销毁线程属性
    ret = pthread_attr_destroy(&attr);
    if (0 != ret) {
        printf("线程属性销毁失败。\n");
        return 1;
    }

    return 0;
}

运行结果:

Linux线程之线程的创建、属性、回收、退出、取消方式

(3)线程栈大小获取和设置

#include<pthread.h>

int pthread_attr_setstacksize(pthread_attr_t* attr, size_t stacksize);
/*
功能:
    设置线程栈大小。
参数:
    attr:线程属性结构体;
    stacksize:线程栈大小;
返回值:China编程
    成功:0;
    失败:错误码
*/

int pthread_attr_getstacksize(const pthread_attr_t* attr, size_t* stacksize);
/*
功能:
    获取线程栈大小。
参数:
    attr:线程属性结构体指针;
    stacksize:返回的线程栈大小;
返回值:
    成功:0;
    失败:非0错误码
*/

4. 线程的回收

(1)pthread_join函数

主线程回收线程资源,会阻塞。

#include<pthread.h>

int pthread_join(pthread_t thread, void** retval);
/*
功能:
    类似于wait()函数。等待线程thread结束,回收线程资源;若线程已结束,则会立即返回。
参数:
    thread:等待回收的线程号;
    retval:存储进程退出状态的指针的地址;
返回值:
    成功:0
    失败:非0错误码
*/

pthread_join示例:

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

void* func() {
    printf("子线程开始执行...\n");
    sleep(3);
    printf("子线程结束执行...\n");
    return (void*)0x3;
}

int main(int argc, const char* argv[]) {

    pthread_t tid;
    int ret = -1;
    void* retp = NULL;

    memset(&tid, 0, sizeof(tid));

    // 创建线程
    ret = pthread_create(&tid, NULL, func, NULL);
    if (0 != ret) {
        printf("线程创建失败.\n");
        return 1;
    }

    printf("主线程执行...\n");

    // 等待线程结束 pthread_join会阻塞
    ret = pthread_join(tid, &retp);
    if (0 != ret) {
        printf("线程join失败.\n");
        return 1;
    }

    printf("retp: %p\n", retp);
    printf("主线程退出...\n");

    return 0;
}

运行结果:

Linux线程之线程的创建、属性、回收、退出、取消方式

(2)pthread_detach函数

内核回收线程资源,不会阻塞。

#include<pthread.h>

int pthread_detach(pthread_t thread);
/*
功能:
    使线程thread与当前进程分离,之后线程结束后的资源回收由内核完成,因此不会阻塞。
参数:
    thread:线程号;
返回值:
    成功:0
    失败:非0
*/

 pthread_detach示例:

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

void* func(void* arg) {
    printf("子线程开始...\n");
    for (int i = 0; i < 3;++i) {
        sleep(1);
        printf("子线程工作%ds\n", i);
    }
    printf("子线程结束...\n");
    return NULL;
}

int main(int argc, const char* argv[]) {

    int ret = -1;
    pthread_t tid = -1;

    // 创建线程
    ret = pthread_create(&tid, NULL, func, NULL);
    if (0 != ret) {
        printf("线程创建失败.\n");
        return 1;
    }

    // 设置线程分离
    ret = pthread_detach(tid);
    if (0 != ret) {
        printf("线程分离失败.\n");
        return 1;
    }

    printf("主线程:按回车键退出..\n");
    getchar();

    return 0;
}

运行结果:

线程设置为detach状态,主线程不必阻塞等待回收子线程资源,而是由内核完成。

Linux线程之线程的创建、属性、回收、退出、取消方式

5. 线程的退出

若在线程中用exit函数退出,则导致整个进程退出,而非退出这一个线程。

如下三者可在不结束整个进程的情况下结束线程:

a)线程从执行函数中返回;

b)线程调用pthread_exit退出线程;

c)线程被同一进程中的其它线程取消。

pthread_exit函数

#include<pthread.h>

void pthread_exit(void* retval);
/*
功能:
    退出调用线程。
参数:
    retval:存储线程退出状态的指针。
*/

6. 线程的取消

pthread_cancel函数

#include<pthread.h>

int pthread_cancel(pthread_t thread);
/*
功能:
    杀死线程thread;
参数:
    thread:目标线程ID;
返回值:
    成功:0
    失败:非0错误码。
*/

pthread_cancel示例:

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

void* func(void* arg) {
    printf("子线程开始...\n");
    for (int i = 0; i < 5;++i) {
        sleep(1);
        printf("子线程工作%ds\n", i);
    }
    printf("子线程结束...\n");
    pthread_exit(NULL);
}

int main(int argc, const char* argv[]) {

    int ret = -1;
    pthread_t tid = -1;

    // 创建线程
    ret = pthread_create(&tid, NULL, func, NULL);
    if (0 != ret) {
        printf("线程创建失败.\n");
        return 1;
    }

    // 设置线程分离
    ret = pthread_detach(tid);
    if (0 != ret) {
        printf("线程分离失败.\n");
        return 1;
    }

    sleep(3);
    pthread_cancel(tid);  // 杀死线程

    printf("主线程:按回车键退出..\n");
    getchar();

    return 0;
}

运行结果:

Linux线程之线程的创建、属性、回收、退出、取消方式

7. 线程使用注意事项

(1)主线程退出,而其余线程不退出,主线程应调用pthread_exit

(2)避免僵尸线程方式:

  • a)pthread_join;
  • b)pthread_detach;
  • c)pthread_create前设置线程分离属性;

(3)malloc和mmap申请的内存可被其他线程释放;

(4)避免在多线程中fork,除非马上使用exec;

(5)避免多线程中使用信号机制。

总结

以上为个人经验,希望能给大家一个参考,也希望大家多多支持编程China编程(www.chinasem.cn)。

这篇关于Linux线程之线程的创建、属性、回收、退出、取消方式的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1155363

相关文章

Linux join命令的使用及说明

《Linuxjoin命令的使用及说明》`join`命令用于在Linux中按字段将两个文件进行连接,类似于SQL的JOIN,它需要两个文件按用于匹配的字段排序,并且第一个文件的换行符必须是LF,`jo... 目录一. 基本语法二. 数据准备三. 指定文件的连接key四.-a输出指定文件的所有行五.-o指定输出

Linux jq命令的使用解读

《Linuxjq命令的使用解读》jq是一个强大的命令行工具,用于处理JSON数据,它可以用来查看、过滤、修改、格式化JSON数据,通过使用各种选项和过滤器,可以实现复杂的JSON处理任务... 目录一. 简介二. 选项2.1.2.2-c2.3-r2.4-R三. 字段提取3.1 普通字段3.2 数组字段四.

Linux kill正在执行的后台任务 kill进程组使用详解

《Linuxkill正在执行的后台任务kill进程组使用详解》文章介绍了两个脚本的功能和区别,以及执行这些脚本时遇到的进程管理问题,通过查看进程树、使用`kill`命令和`lsof`命令,分析了子... 目录零. 用到的命令一. 待执行的脚本二. 执行含子进程的脚本,并kill2.1 进程查看2.2 遇到的

JDK21对虚拟线程的几种用法实践指南

《JDK21对虚拟线程的几种用法实践指南》虚拟线程是Java中的一种轻量级线程,由JVM管理,特别适合于I/O密集型任务,:本文主要介绍JDK21对虚拟线程的几种用法,文中通过代码介绍的非常详细,... 目录一、参考官方文档二、什么是虚拟线程三、几种用法1、Thread.ofVirtual().start(

Java AOP面向切面编程的概念和实现方式

《JavaAOP面向切面编程的概念和实现方式》AOP是面向切面编程,通过动态代理将横切关注点(如日志、事务)与核心业务逻辑分离,提升代码复用性和可维护性,本文给大家介绍JavaAOP面向切面编程的概... 目录一、AOP 是什么?二、AOP 的核心概念与实现方式核心概念实现方式三、Spring AOP 的关

Java 虚拟线程的创建与使用深度解析

《Java虚拟线程的创建与使用深度解析》虚拟线程是Java19中以预览特性形式引入,Java21起正式发布的轻量级线程,本文给大家介绍Java虚拟线程的创建与使用,感兴趣的朋友一起看看吧... 目录一、虚拟线程简介1.1 什么是虚拟线程?1.2 为什么需要虚拟线程?二、虚拟线程与平台线程对比代码对比示例:三

k8s按需创建PV和使用PVC详解

《k8s按需创建PV和使用PVC详解》Kubernetes中,PV和PVC用于管理持久存储,StorageClass实现动态PV分配,PVC声明存储需求并绑定PV,通过kubectl验证状态,注意回收... 目录1.按需创建 PV(使用 StorageClass)创建 StorageClass2.创建 PV

Linux云服务器手动配置DNS的方法步骤

《Linux云服务器手动配置DNS的方法步骤》在Linux云服务器上手动配置DNS(域名系统)是确保服务器能够正常解析域名的重要步骤,以下是详细的配置方法,包括系统文件的修改和常见问题的解决方案,需要... 目录1. 为什么需要手动配置 DNS?2. 手动配置 DNS 的方法方法 1:修改 /etc/res

Linux创建服务使用systemctl管理详解

《Linux创建服务使用systemctl管理详解》文章指导在Linux中创建systemd服务,设置文件权限为所有者读写、其他只读,重新加载配置,启动服务并检查状态,确保服务正常运行,关键步骤包括权... 目录创建服务 /usr/lib/systemd/system/设置服务文件权限:所有者读写js,其他

Linux下利用select实现串口数据读取过程

《Linux下利用select实现串口数据读取过程》文章介绍Linux中使用select、poll或epoll实现串口数据读取,通过I/O多路复用机制在数据到达时触发读取,避免持续轮询,示例代码展示设... 目录示例代码(使用select实现)代码解释总结在 linux 系统里,我们可以借助 select、