news 2026/3/24 2:49:32

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/3/15 14:43:38

零基础也能懂的电源管理系统概述

电源管理&#xff1a;不只是“供电”那么简单你有没有想过&#xff0c;为什么你的手机能一边充电、一边快充、一边还能正常运行&#xff1f;为什么一块小小的电池能让智能手表连续工作好几天&#xff1f;为什么有些设备一开机就死机&#xff0c;而另一些却稳定如初&#xff1f;…

作者头像 李华
网站建设 2026/3/21 6:31:19

LCD12864并行模式新手教程:基础接线与测试

从零开始玩转 LCD12864&#xff1a;并行驱动实战全记录你有没有遇到过这样的情况&#xff1f;花几十块买了一块看起来挺“高级”的图形屏&#xff0c;接口密密麻麻&#xff0c;接上单片机后却只看到一片黑——既没字也没图&#xff0c;连个光标都不闪。别急&#xff0c;这几乎是…

作者头像 李华
网站建设 2026/3/15 2:24:33

最大似然估计简介

原文&#xff1a;towardsdatascience.com/introduction-to-maximum-likelihood-estimates-7e37f83c6757 简介 最大似然估计&#xff08;MLE&#xff09;是一种基本方法&#xff0c;它使任何机器学习模型都能从可用数据中学习独特的模式。在这篇博客文章中&#xff0c;我们将通…

作者头像 李华
网站建设 2026/3/15 18:45:29

Qwen3-VL浏览GitHub镜像库查找最新AI项目

Qwen3-VL 浏览 GitHub 镜像库查找最新 AI 项目 在多模态 AI 技术飞速演进的今天&#xff0c;开发者面临的不再是“有没有模型可用”&#xff0c;而是“如何快速试用、验证并集成前沿能力”。传统方式下&#xff0c;下载百亿参数模型动辄耗费数小时&#xff0c;环境配置复杂、依…

作者头像 李华
网站建设 2026/3/20 15:32:40

STLink驱动安装操作指南:适用于Windows系统

STLink驱动安装全攻略&#xff1a;从零搞定Windows下的调试连接 在STM32开发的世界里&#xff0c;你可能写过无数行代码、调通过复杂的外设驱动&#xff0c;但最让人抓狂的往往不是程序逻辑&#xff0c;而是—— 电脑连不上STLink调试器 。 插上开发板&#xff0c;打开IDE&…

作者头像 李华
网站建设 2026/3/19 21:20:41

零基础也能懂的nrf52832的mdk下载程序教程

从零开始玩转nRF52832&#xff1a;Keil MDK下载程序全解析&#xff0c;不只是“点一下”那么简单 你有没有过这样的经历&#xff1f; 明明代码写好了&#xff0c;工程也编译通过了&#xff0c;信心满满地点击 Keil 的“Download”按钮&#xff0c;结果弹出一串红字&#xff1…

作者头像 李华