news 2026/5/29 21:29:31

Linux学习日记18:线程的分离

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
Linux学习日记18:线程的分离

一、前言

前面我们学习了线程的一些基础知识,学习了线程的创建与使用,今天我们来学习线程的分离与同步。

二、线程分离

2.1、函数原型

函数原型如下:

#include <pthread.h> int pthread_detach(pthread_t thread);

参数:thread:设置为分离态的线程 ID(用户态 TID,由 pthread_create() 返回的 ID)。

返回值:成功:返回0;失败:返回对应的错误码。

2.2、工作原理

pthread_detach()本质是修改线程的分离状态属性,通知内核:该线程终止时,无需保留资源等待pthread_join(),自动回收资源;具体执行流程如下:

1、用户态方面:pthread库修改该线程对应的属性结构体(pthread_attr_t)中detachstate字段为PTHREAD_CREATE_DETACHED;

2、内核态层面: 库通过系统调用通知内核,标记该线程的 task_struct(任务结构体)为 “分离态”;

3、线程终止时:内核检测到分离态标记,会立即清理线程的栈、寄存器、TID 等资源,无需主线程干预。

注:分离操作不影响线程的执行,仅改变线程终止后的资源回收规则

2.3、典型示例

1、先创建再detach

先创建一个pthread_detach.c文件,然后输入以下代码:

#include <stdio.h> #include <pthread.h> #include <unistd.h> #include <string.h> void *myfun(void *arg) { printf("child pthread id is %ld\n",pthread_self()); for(int i=0;i<5;i++) { printf("children i =%d\n",i); sleep(1); } return NULL;//子线程结束,自动回收资源并退出 } int main() { pthread_t pthid; int ret; ret = pthread_create(&pthid,NULL,myfun,NULL);//线程创建 pthread_detach(pthid);//线程分离 if(ret != 0) { printf("error number is %d\n",ret); printf("%s\n",strerror(ret)); } printf("parent pthread id is %ld\n",pthread_self()); for(int i=0;i<5;i++) { printf("parent i = %d\n",i); sleep(1); } sleep(2);//休眠2s,防止主线程快速退出导致进程退出 return 0; }

使用gcc编译器进行编译,运行结果如下:

可以看到,主线程并不会等待子进程,子进程在后台跑,子线程自动释放资源,这就是线程分离的目的。

2、创建时直接设置为分离态

有时不想手动设置分离态,可以让线程创建时直接分离,具体代码如下:

#include <stdio.h> #include <pthread.h> #include <unistd.h> #include <string.h> void *myfun(void *arg) { printf("child pthread id is %ld\n",pthread_self()); for(int i=0;i<5;i++) { printf("children i =%d\n",i); sleep(1); } return 0; } int main() { pthread_t pthid; pthread_attr_t attr; //init pthread_attr_init(&attr);//初始化线程属性对象 //set pthread_attr_setdetachstate(&attr,PTHREAD_CREATE_DETACHED);//设置线程为分离态 //set2 int ret; ret = pthread_create(&pthid,&attr,myfun,NULL);//创建线程,并使其天生就是分离状态 pthread_detach(pthid); if(ret != 0) { printf("error number is %d\n",ret); printf("%s\n",strerror(ret)); } printf("parent pthread id is %ld\n",pthread_self()); for(int i=0;i<5;i++) { printf("parent i = %d\n",i); sleep(1); } sleep(2); //kill attr pthread_attr_destory(&attr);//销毁属性对象(不影响已经创建的线程) return 0; }

使用gcc编译器进行编译,运行结果如下:

运行结果与第一种方法一样,但是这是创建时直接分离,避免了线程创建失败而导致后续分离操作无效的情况。

2.4、注意事项

1、分离状态不可逆:线程一旦通过 pthread_detach() 设为分离态,无法再改回joinable状态。如果后续试图调用 pthread_join(),会直接返回 EINVAL 错误。

2、调用时机越早越好,避免线程提前终止:若主线程创建子线程后,还没来得及调用 pthread_detach(),子线程就已经终止,此时调用 pthread_detach() 会返回 ESRCH(找不到线程);

3、不能对已终止的线程调用pthread_detach():线程终止后,其 TID 可能被内核复用(分配给新创建的线程),此时调用 pthread_detach() 可能误操作新线程,导致不可预期的问题。

4、分离线程崩溃仍会导致进程终止:pthread_detach() 仅改变资源回收方式,不改变线程与进程的资源共享关系:如果分离线程触发段错误、除零等异常,仍会发送信号终止整个进程。

5、主线程提前退出仍会杀死分离线程:分离线程只是 “终止后自动回收资源”,但如果主线程直接return/exit导致进程终止,所有分离线程仍会被强制杀死(需主线程 pthread_exit()或延时等待)。

6、不能同时调用 pthread_join() 和 pthread_detach():若先调用 pthread_join()阻塞等待线程,此时再调用 pthread_detach() 会返回 EINVAL;若先调用 pthread_detach() 分离线程,再调用 pthread_join()也会返回 EINVAL;

7、主线程无需分离(无意义):主线程的 TID 等于进程 PID,分离主线程不会改变 “主线程退出导致进程终止” 的规则,因此对主线程调用 pthread_detach()无实际意义,还可能返回 EINVAL 错误。

2.5、两种创建方法对比

二者对比如下:

特性pthread_detach()(动态分离)pthread_attr_setdetachstate()(静态创建)
调用时机线程创建后任意时间(需线程未终止)线程创建前(初始化属性时)
灵活性高(运行时决定是否分离)低(创建前确定,无法动态调整)
代码复杂度低(无需初始化属性结构体)高(需初始化 / 销毁属性结构体)
风险可能因线程提前终止导致调用失败无此风险(创建时直接设为分离态)
适用场景运行时才确定是否分离的场景提前确定无需等待的后台线程
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/5/27 1:46:55

PHPBrew自定义任务开发实战:从入门到精通

想要让PHPBrew真正成为你的专属开发利器吗&#xff1f;自定义任务就是那把钥匙。通过它&#xff0c;你可以让PHP环境管理变得随心所欲&#xff0c;不再受限于标准功能。今天我们就来聊聊如何玩转PHPBrew自定义任务开发。 【免费下载链接】agent-framework A framework for buil…

作者头像 李华
网站建设 2026/5/29 17:10:39

PACKAGER.EXE:Windows工具包的终极对象安装包解决方案

PACKAGER.EXE&#xff1a;Windows工具包的终极对象安装包解决方案 【免费下载链接】packager.exe资源下载介绍 PACKAGER.EXE是一款专为微软Windows操作系统设计的实用工具&#xff0c;特别适用于解决Windows XP系统中Word文档内嵌对象无法打开的问题。通过该工具&#xff0c;用…

作者头像 李华
网站建设 2026/5/30 13:49:50

Java:serialVersionUID

serialVersionUID是 Java 中用于序列化机制的一个特殊字段&#xff0c;它是一个 static final long 类型的标识符&#xff0c;用于标识可序列化类的版本。‌ 作用与原理当一个类实现 Serializable 接口时&#xff0c;Java 序列化机制会为该类关联一个版本号&#xff08;即 seri…

作者头像 李华
网站建设 2026/5/28 16:05:24

[特殊字符] 别只盯着跑分!开发者已偷偷爱上 Gemini 3,GPT-5.2 还能用情怀留住他们吗?一场关于钱、效率和生态的血战!

咱们聊点更接地气的&#xff1a;谁才是真正的 AI “印钞机”&#xff1f; 跑分、参数、论文&#xff0c;这些都是给投资人和媒体看的。但对于每天用 API 接口、用 AI 赚钱的开发者和创业公司来说&#xff0c;他们只看三点&#xff1a;性能/价格比&#xff08;性价比&#xff09…

作者头像 李华
网站建设 2026/5/29 12:20:38

手把手教你用 ScreenToGif 录制屏幕并生成 GIF 动图

在写技术文档、提交 Bug 反馈、制作教学视频或分享软件操作技巧时&#xff0c;一张会“动”的 GIF 图往往比千言万语更直观有效。而 ScreenToGif 正是 Windows 平台上最受欢迎的免费、开源、无广告的屏幕录制工具之一&#xff0c;能让你快速将桌面操作录制成高质量 GIF 动图。接…

作者头像 李华
网站建设 2026/5/30 15:17:12

自学嵌入式day29,进程

以下是针对您提供的关于进程的笔记内容的整理和解释。我将以清晰的结构帮助您逐步理解这些操作系统概念。内容基于标准的操作系统知识&#xff0c;确保真实可靠。1. 进程的含义进程是一个程序在执行过程中的动态实体&#xff0c;它涉及内存资源的分配和 CPU 的调度。进程的核心…

作者头像 李华