news 2026/5/3 17:05:34

Linux 线程编程 - 线程属性:脱离状态 + 调度策略

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
Linux 线程编程 - 线程属性:脱离状态 + 调度策略

作为编程初学者,学完基础的线程创建(pthread_create)后,遇到了 “批量配置线程特征” 的场景 —— 比如让线程自动释放资源、调整线程调度优先级,这时线程属性(pthread_attr_t)就派上用场了!整理核心知识点和实战代码如下

一、核心特点

1. 适用场景

  • 批量创建线程,统一配置线程特征(脱离状态、调度策略、优先级等);
  • 避免线程创建时硬编码配置,提升代码灵活性和可维护性;
  • 需精细化控制线程运行规则(如实时线程的调度优先级)。

2. 核心逻辑

线程属性本身不控制线程运行,而是为 pthread_create 提供统一的配置模板;核心流程:初始化属性 → 设置特征属性 → 创建线程 → 销毁属性;属性销毁后不影响已创建的线程,仅释放属性结构体资源。

二、关键接口

接口功能关键说明
pthread_attr_init动态初始化线程属性attr(属性变量指针);成功返回 0,失败返回错误码
pthread_attr_destroy销毁线程属性释放属性资源;销毁后可重新初始化;成功返回 0
pthread_attr_setdetachstate设置线程脱离状态可选值:① PTHREAD_CREATE_DETACHED(脱离态,线程结束自动释放资源,主线程不可 pthread_join)② PTHREAD_CREATE_JOINABLE(默认,主线程可 pthread_join)
pthread_attr_getdetachstate获取线程脱离状态存入 detachstate 指针,用于校验属性配置结果
pthread_attr_setschedpolicy设置调度策略可选值:① SCHED_OTHER(默认,普通调度,无实时性)② SCHED_FIFO(实时 FIFO,需 sudo 运行)③ SCHED_RR(实时轮转,需 sudo 运行)
pthread_attr_getschedpolicy获取调度策略存入 policy 指针,校验调度策略配置
sched_get_priority_max/min获取优先级范围配合调度策略使用;SCHED_OTHER 优先级固定为 0,SCHED_FIFO/RR 为 1~99
pthread_attr_setschedparam设置线程优先级需传入 sched_param 结构体(含 sched_priority);仅对实时策略有效

重点:脱离状态核心逻辑

  • 脱离态(DETACHED):线程结束后自动释放资源,主线程无需 pthread_join,适合批量创建的无关联线程;
  • 可连接态(JOINABLE):默认状态,主线程需 pthread_join 回收资源,否则会产生 “僵尸线程”。

三、实战代码

示例 1:脱离状态属性(核心常用)

模拟创建 “脱离态” 线程,线程结束自动释放资源,主线程通过共享变量等待线程结束:

#include <stdio.h> #include <unistd.h> #include <stdlib.h> #include <pthread.h> int finished = 0; // 共享变量判断线程是否结束 void* thread_function(void* arg) { printf("线程开始运行...\n"); int count = (int)arg; for (int i = 0; i < count; i++) { printf("线程正在工作,第%d次\n", i + 1); sleep(1); } printf("线程即将关闭.\n"); finished = 1; pthread_exit(NULL); } int main() { pthread_attr_t thread_attr; int res; // 1. 初始化线程属性 res = pthread_attr_init(&thread_attr); if (res != 0) { perror("pthread_attr_init failed"); exit(1); } // 2. 设置为脱离状态(核心) res = pthread_attr_setdetachstate(&thread_attr, PTHREAD_CREATE_DETACHED); if (res != 0) { perror("Setting detached attribute failed"); exit(EXIT_FAILURE); } // 3. 创建线程(传入配置好的属性) pthread_t a_thread; int n = 3; // 线程工作次数 res = pthread_create(&a_thread, &thread_attr, thread_function, (void*)n); if (res != 0) { perror("Thread creation failed"); exit(EXIT_FAILURE); } // 4. 销毁属性(不影响已创建线程) pthread_attr_destroy(&thread_attr); // 脱离状态下无法pthread_join,用共享变量等待 while (!finished) { printf("等待线程结束...\n"); sleep(1); } printf("线程已结束\n"); return 0; }

示例 2:调度策略 + 优先级属性

配置线程调度策略为默认(SCHED_OTHER),并设置优先级(仅演示流程,实时策略需 sudo 运行):

#include <stdio.h> #include <unistd.h> #include <stdlib.h> #include <pthread.h> #include <sched.h> int finished = 0; void* thread_function(void* arg) { printf("线程开始运行...\n"); int count = (int)arg; for (int i = 0; i < count; i++) { printf("线程正在工作,第%d次\n", i + 1); sleep(1); } printf("线程即将关闭.\n"); finished = 1; pthread_exit(NULL); } int main() { pthread_attr_t thread_attr; int res; // 1. 初始化属性 res = pthread_attr_init(&thread_attr); if (res != 0) { perror("pthread_attr_init failed"); exit(1); } // 2. 设置脱离状态 res = pthread_attr_setdetachstate(&thread_attr, PTHREAD_CREATE_DETACHED); if (res != 0) { perror("Setting detached attribute failed"); exit(EXIT_FAILURE); } // 3. 设置调度策略(SCHED_OTHER/SCHED_FIFO/SCHED_RR) res = pthread_attr_setschedpolicy(&thread_attr, SCHED_OTHER); if (res != 0) { perror("pthread_attr_setschedpolicy failed"); exit(1); } // 4. 获取优先级范围并设置 int max_priority = sched_get_priority_max(SCHED_OTHER); int min_priority = sched_get_priority_min(SCHED_OTHER); printf("SCHED_OTHER优先级范围:%d ~ %d\n", min_priority, max_priority); struct sched_param scheduling_value; scheduling_value.sched_priority = min_priority; // SCHED_OTHER仅能设0 res = pthread_attr_setschedparam(&thread_attr, &scheduling_value); if (res != 0) { perror("pthread_attr_setschedparam failed"); exit(1); } // 5. 创建线程 pthread_t a_thread; int n = 3; res = pthread_create(&a_thread, &thread_attr, thread_function, (void*)n); if (res != 0) { perror("Thread creation failed"); exit(EXIT_FAILURE); } // 6. 销毁属性 pthread_attr_destroy(&thread_attr); // 等待线程结束 while (!finished) { printf("等待线程结束...\n"); sleep(1); } printf("线程已结束\n"); return 0; }

四、编译运行

线程属性依赖 pthread 库,编译时必须加-lpthread

# 编译脱离状态示例 gcc thread_attr_detach.c -o thread_attr_detach -lpthread # 运行 ./thread_attr_detach # 编译调度策略示例(实时策略需sudo) gcc thread_attr_sched.c -o thread_attr_sched -lpthread # 普通策略运行 ./thread_attr_sched # 实时策略(如SCHED_FIFO)需sudo sudo ./thread_attr_sched

五、踩坑

  1. 初始化属性前使用:未调用 pthread_attr_init 直接设置属性,导致程序崩溃或配置无效;
  2. 脱离态线程调用 pthread_join:pthread_join 返回错误,无法等待线程结束;
  3. 实时调度策略未用 sudo 运行:SCHED_FIFO/SCHED_RR 配置失败,无权限修改调度策略;
  4. SCHED_OTHER 设置非 0 优先级:优先级配置无效,SCHED_OTHER 仅支持优先级 0;
  5. 属性销毁后重复使用:销毁后未重新初始化直接设置属性,导致配置失败;
  6. 忽略接口返回值:未校验 pthread_attr_setXXX 的返回值,配置失败后无感知。

六、总结

线程属性是 “线程配置的统一模板”,核心价值在于批量、灵活配置线程特征

  • 脱离状态:解决 “僵尸线程” 问题,批量创建线程时优先使用 DETACHED;
  • 调度策略:普通场景用 SCHED_OTHER,实时场景(如工控、音视频)用 SCHED_FIFO/RR;
  • 核心原则:属性仅影响创建时的线程,销毁不影响已运行线程,需配合互斥锁 / 条件变量完成复杂同步。

对比直接传 NULL 给 pthread_create(默认属性),线程属性让线程配置更清晰、可维护,是 Linux 线程进阶编程的必备知识点。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/25 11:53:14

嵌入式工控主板USB Serial驱动下载实战演示

嵌入式工控主板上USB转串口驱动的落地实战&#xff1a;从芯片到系统&#xff0c;打通工业通信“最后一公里”你有没有遇到过这样的场景&#xff1f;一台崭新的嵌入式工控主板通电后&#xff0c;连接传感器却收不到数据&#xff1b;调试线插上去&#xff0c;PC端设备管理器里只显…

作者头像 李华
网站建设 2026/4/26 11:54:46

Qwen3-VL与Dify集成方案探索:构建企业级AI应用

Qwen3-VL与Dify集成方案探索&#xff1a;构建企业级AI应用 在企业智能化转型的浪潮中&#xff0c;一个核心挑战正日益凸显——如何让AI真正“看懂”现实世界&#xff1f;传统大模型擅长处理文本&#xff0c;却对图像、界面截图甚至一份带图表的PDF束手无策。而今天&#xff0c;…

作者头像 李华
网站建设 2026/5/1 3:40:58

产品质量可靠的LED工矿灯厂家有哪些?

“选择LED工矿灯厂家&#xff0c;找的不只是产品&#xff0c;更是长达十年甚至更久的照明解决方案与工程伙伴。”当您作为工厂老板、项目负责人&#xff0c;或基建采购决策者&#xff0c;被问及“产品质量可靠的LED工矿灯厂家有哪些”时&#xff0c;真正的潜台词往往是&#xf…

作者头像 李华
网站建设 2026/4/29 15:39:29

对象存储OSS存放Sonic原始素材与成品视频链接

对象存储OSS存放Sonic原始素材与成品视频链接 在短视频内容爆炸式增长的今天&#xff0c;企业对高效、低成本生成数字人视频的需求愈发迫切。传统依赖3D建模和高性能渲染的工作流不仅周期长、成本高&#xff0c;还难以适应快速迭代的内容生产节奏。而腾讯联合浙江大学推出的轻量…

作者头像 李华
网站建设 2026/4/22 5:36:06

【Java进阶】面向对象编程第一站:深入理解类、对象与封装前言

大家好&#xff0c;我是夏幻灵&#xff0c;今天带大家跨入Java编程的核心领域——面向对象编程。这是Java语言的灵魂所在。本篇我们将攻克最基础也是最重要的两个概念&#xff1a;“类”与“对象”&#xff0c;并深入探讨面向对象三大特征之一的“封装”。一、 类与对象的本质在…

作者头像 李华
网站建设 2026/5/2 7:21:57

零基础入门:搭建STM32 + TouchGFX开发环境

从零开始&#xff1a;手把手搭建 STM32 TouchGFX 图形开发环境 你有没有遇到过这样的场景&#xff1f;项目需要一个带触摸屏的界面&#xff0c;老板说“要好看、要流畅”&#xff0c;可你手里只有一块STM32单片机&#xff0c;连操作系统都没有。这时候&#xff0c;大多数人第…

作者头像 李华