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; ?>
】
相关推荐
- Mac如何修改Hosts文件?(本地开发与屏蔽网站
- php查询数据怎么导出csv_查询结果转csv文件
- 如何使用Golang优化模块引入路径_Golang
- Python 中将 ISO 8601 时间戳转换为
- 如何在Golang中使用replace替换模块_指
- Golang如何测试HTTP中间件_Golang
- mac怎么看硬盘大小_MAC查看磁盘存储空间与文件
- Win11如何设置文件关联 Win11修改特定文件
- Win11开始菜单打不开_修复Windows 11
- Windows10如何彻底关闭自动更新_Win10
- Windows系统被恶意软件破坏后的恢复策略_错误
- C++中的std::shared_from_thi
- LINUX怎么进行文本内容搜索_Linux gre
- 如何使用Golang构建简易投票统计功能_Gola
- Windows音频驱动无声音原因解析_声卡驱动错误
- Win10如何更改网络连接_Windows10以太
- Win11怎么更改输入法顺序_Win11调整语言首
- 如何在Golang中处理云原生事件_使用Event
- c++怎么使用类型萃取type_traits_c+
- VSC怎么在PHP中调试MySQL_数据库交互排查
- Windows服务持续崩溃怎样修复_系统服务保护机
- Windows怎样拦截WPS弹窗广告_Window
- Win11怎么更改计算机名_Windows11系统
- PHP 中如何在函数内持久化修改引用变量的指向
- 如何正确访问 Laravel 模型或对象的属性而非
- Win11如何设置环境变量 Win11添加和修改系
- Go语言中slice追加操作的底层共享机制解析
- php怎么连接数据库_MySQL数据库连接的基础代
- c++输入输出流 c++ cin与cout格式化输
- 如何使用Golang实现微服务事件驱动_使用消息总
- c++ unordered_map怎么用 c++哈
- 如何使用Golang读取日志文件_Golang b
- 如何在Golang中实现RPC异步返回_Golan
- LINUX下如何配置VLAN虚拟局域网_在LINU
- Win11怎么关闭资讯和兴趣_Windows11任
- Windows 11怎么设置默认解压软件_Wind
- Windows10电脑怎么设置虚拟内存_Win10
- Windows蓝屏错误0x0000002C怎么解决
- Win11怎么设置任务栏大小_Windows11注
- Win10怎么关闭自动更新错误重启 Win10策略
- php订单日志怎么在swoole写_php协程sw
- Windows蓝屏错误0x0000001E怎么修复
- Go 中的 := 运算符:类型推导机制与使用边界详
- Linux怎么禁止Root用户远程登录_Linux
- Python如何创建带属性的XML节点
- 如何减少Golang内存碎片化_Golang内存分
- LINUX的SELinux是什么_详解LINUX强
- Linux如何申请SSL免费证书_Linux下Ce
- 如何使用Golang实现聊天室消息存档_存储聊天记
- Win11系统占用空间大怎么办 Win11深度瘦身

QQ客服