c++怎么设置线程优先级与cpu亲和性_c++ 多核处理器性能绑定【指南】
技术百科
穿越時空
发布时间:2026-01-01
浏览: 次 Linux下需用pthread_setschedparam设置线程优先级,仅std::thread对象无效;须root权限或CAP_SYS_NICE能力才能设SCHED_FIFO/RR实时策略,优先级范围1–99;绑定CPU用pthread_setaffinity_np,需检查CPU在线状态及错误码。
Linux 下用 pthread_setschedparam 设置线程优先级
在 Linux 中,C++ 线程(std::thread)底层通常基于 pthread,但标准库不暴露调度参数接口,必须用原生 pthread 函数操作。直接调用 pthread_sets 才能真正生效,仅改
chedparamstd::thread 对象本身无效。
注意:需要 root 权限或 CAP_SYS_NICE 能力才能提升实时优先级(如 SCHED_FIFO),否则会静默失败或返回 EPERM。
- 先用
pthread_self()获取当前线程 ID,再传给pthread_setschedparam - 策略选
SCHED_FIFO或SCHED_RR才支持优先级(SCHED_OTHER的priority必须为 0) - 优先级范围依赖策略:
SCHED_FIFO通常为 1–99,可通过sched_get_priority_min/max(SCHED_FIFO)查
struct sched_param param;
param.sched_priority = 50;
int result = pthread_setschedparam(pthread_self(), SCHED_FIFO, ¶m);
if (result != 0) {
// 检查 errno,常见为 EPERM(权限不足)或 EINVAL(策略/优先级非法)
}
绑定线程到指定 CPU 核心用 pthread_setaffinity_np
pthread_setaffinity_np 是 GNU 扩展,非 POSIX 标准,但 Linux 上稳定可用。它控制线程只能在指定 CPU 核心上运行,避免跨核迁移开销,对低延迟场景关键。
容易踩的坑:CPU 编号从 0 开始,且需确认系统实际核心数(nproc 或 /sys/devices/system/cpu/online),绑到不存在的 CPU 会失败;另外,亲和性设置对子线程不继承,每个线程需单独设。
- 用
cpu_set_t类型构造掩码,CPU_ZERO清空,CPU_SET(2)表示启用 CPU 2 - 调用前确保目标 CPU 处于 online 状态(热插拔可能使某些 core offline)
- 若程序运行时被系统调度器抢占到其他核,说明亲和性未生效——大概率是没正确调用或返回值未检查
cpu_set_t cpuset;
CPU_ZERO(&cpuset);
CPU_SET(1, &cpuset); // 绑定到 CPU 1
int result = pthread_setaffinity_np(pthread_self(), sizeof(cpu_set_t), &cpuset);
if (result != 0) {
// 检查 errno,常见为 EINVAL(CPU 编号越界)或 ESRCH(线程已退出)
}
std::thread 启动后如何安全获取并设置 pthread 层属性
std::thread 不提供公开的 native_handle() 类型定义,但标准允许其返回可转换为 pthread_t 的类型(GCC/Clang 下就是 pthread_t)。只要不依赖 ABI 细节,可直接转型使用。
关键点:必须在线程已启动、但尚未结束时调用,否则 native_handle() 可能为无效值;且不能在 join() 或 detach() 后使用。
- 用
auto handle = t.native_handle()获取句柄,再转为pthread_t - 优先级与亲和性设置建议放在新线程函数入口处,而非主线程中调用——避免竞态和时序问题
- 若线程函数是 lambda,捕获变量需注意生命周期,避免设置时线程已退出
实时调度 + CPU 绑定组合使用的典型陷阱
两者一起用时,性能收益明显,但错误配置反而导致卡死或吞吐暴跌。最常被忽略的是:实时线程若无主动让出(如 usleep、nanosleep、阻塞 I/O),会饿死其他进程,尤其当它跑满单核且优先级高于所有普通线程时。
- 不要把多个高优先级线程绑到同一 CPU 核——它们会互相抢占,失去实时性保障
- 避免在实时线程中调用可能分配内存、加锁或触发 page fault 的操作(如
std::cout、malloc) - 测试阶段务必用
taskset -c N ./a.out验证亲和性是否生效,再用chrt -f 50 ./a.out验证调度策略,比代码内调试更直观
真实场景里,优先级数字和 CPU 编号不是写死的常量,应从环境变量或配置文件读取,并做存在性校验——这点几乎所有人一开始都会忽略。
# 的是
# 放在
# 能在
# 要把
# 多个
# 不存在
# 绑定
# 能使
# 配置文件
# 再用
# auto
# linux
# 环境变量
# 对象
# c++
# 标准库
# 接口
# 线程
# gnu
# 继承
# Thread
# 处理器
# Lambda
# 句柄
# 主线程
# 常量
相关栏目:
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
AI推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
SEO优化<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
技术百科<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
谷歌推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
百度推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
网络营销<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
案例网站<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
精选文章<?muma echo $count; ?>
】
相关推荐
- Win11怎么关闭触控板_Win11笔记本禁用触摸
- Python文本编码与解码_跨平台解析说明【指导】
- Python对象比较与排序_集合使用说明【指导】
- Windows 11如何开启文件夹加密(EFS)_
- 如何使用Golang实现基本类型比较_Golang
- Mac如何与安卓手机传文件_Mac和Android
- c# Task.ConfigureAwait(tr
- 如何使用Golang安装API文档生成工具_快速生
- C++ STL算法库怎么用?C++常用算法函数(s
- Win11如何添加/删除输入法 Win11切换中英
- Win11如何更新显卡驱动 Win11检查和安装设
- php中::能访问全局变量吗_全局作用域与类作用域
- Win10闹钟铃声怎么自定义 Win10闹钟自定义
- Win11怎么更改任务栏颜色_Windows11个
- Win10系统映像怎么恢复 Win10使用系统映像
- php打包exe后无法读取环境变量_变量配置方法【
- Linux怎么设置磁盘配额_Linux系统Quot
- 如何高效识别并拦截拼接式恶意域名 spam
- Win11怎么忘记WiFi网络_Win11删除已保
- 如何在Golang中实现CI/CD流水线自动化测试
- Win11怎么查看电脑配置_Win11硬件配置详细
- Win11怎么更改鼠标指针方案_Windows11
- php条件判断怎么写_ifelse和switchc
- Win11怎么设置触控板手势_Windows11三
- Go语言中CookieJar的持久化机制解析:内存
- 如何在Golang中实现服务熔断与限流_Golan
- Mac怎么设置登录项_Mac管理开机自启动程序【教
- Mac的访达(Finder)怎么用_Mac文件管理
- Win11怎么关闭用户账户控制UAC_Window
- 如何在Golang中使用log包输出不同级别日志_
- Win11怎么更改系统语言_Win11中文语言包下
- C++如何使用std::optional?(处理可
- Win11 C盘满了怎么清理 Win11磁盘清理和
- Go语言中slice追加操作的底层共享机制解析
- 如何使用Golang开发基础文件下载功能_Gola
- LINUX怎么进行文本内容搜索_Linux gre
- 为什么Go建议使用error接口作为错误返回_Go
- Windows系统文件被保护机制阻止怎么办_权限不
- c# 如何用c#实现一个支持优先级的任务队列
- Win11怎么检查TPM2.0模块_Windows
- Win11怎么关闭通知消息_屏蔽Windows 1
- php怎么下载安装后测试是否成功_简单脚本验证方法
- c++ std::atomic如何保证原子性 c+
- Windows如何设置登录时的欢迎屏幕背景?(锁屏
- 如何使用正则表达式提取以编号开头、后接多个注解的逻
- Drupal 中渲染节点时出现 HTML 标签嵌套
- 如何使用Golang encoding/json解
- Windows10系统怎么查看运行时间_Win10
- c++获取当前时间戳_c++ time函数使用详解
- Windows 11如何查看系统激活密钥_Wind

QQ客服