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; ?>
】
相关推荐
- 如何在Golang中操作嵌套切片指针_Golang
- 如何在 PHP 单元测试中正确模拟带方法的图像处理
- Win10怎么卸载剪映_Win10彻底卸载剪映方法
- LINUX的SELinux是什么_详解LINUX强
- Win10电脑怎么设置休眠快捷键_Windows1
- Win11怎么退出高对比度模式_Win11取消反色
- c++中如何使用std::variant_c++1
- Windows10系统怎么查看防火墙状态_Win1
- Win11怎么更改电脑密码_Windows 11修
- Win10如何卸载预装Edge扩展_Win10卸载
- C++中的协变与逆变是什么?C++函数指针与返回类
- C#如何在一个XML文件中查找并替换文本内容
- php查询数据怎么分组_groupby分组查询配合
- 如何正确访问 Laravel 模型或对象的属性而非
- mac本地php环境如何开启curl_curl扩展
- php下载安装后swoole扩展怎么安装_异步框架
- VSC怎么创建PHP项目_从零开始搭建项目的步骤【
- Linux怎么禁止Root用户远程登录_Linux
- PythonGIL机制理解_多线程限制解析【教程】
- Win10如何卸载Skype_Win10卸载Sky
- 如何使用Golang实现Web表单数据绑定_自动映
- 如何使用Golang recover捕获panic
- Win11怎么设置默认图片查看器_Windows1
- Mac怎么设置鼠标滚动速度_Mac鼠标设置详细参数
- 手机php怎么转mp4_手机端php文件转mp4a
- 本地php环境打开php文件直接下载_浏览器解析p
- Win11时间格式怎么改成12小时制 Win11时
- php485能和物联网模块通信吗_php485对接
- 如何使用正则表达式批量替换重复的“-”模式为固定字
- 如何在Golang中实现文件下载_Golang文件
- Win11怎么恢复旧版开始菜单_通过软件还原Win
- mac怎么安装字体_MAC添加第三方字体与字体册管
- Dapper的Execute方法的返回值是什么意思
- 如何使用Golang处理静态文件缓存_提高页面加载
- php485返回空数组怎么回事_php485数据接
- Win11怎么开启剪贴板历史记录_Windows1
- Win10如何关闭安全中心所有通知 Win10禁用
- Win10路由器怎么隐藏ssid Win10隐藏w
- 如何在 Go 结构体中正确初始化 map 字段
- Windows如何拦截2345弹窗广告_Windo
- Windows 10怎么录屏_Windows 10
- 如何在 Windows 11 中使用 AlomWa
- Win11怎么设置任务栏对齐方式_Windows1
- c++ reinterpret_cast怎么用 c
- php怎么连接数据库_MySQL数据库连接的基础代
- mac怎么打开终端_MAC终端Terminal使用
- Win10怎样清理C盘浏览器缓存_Win10清理浏
- Win10怎样设置闹钟贪睡时间 Win10闹钟贪睡
- 如何使用Golang匿名函数_快速定义临时函数逻辑
- php485函数怎么捕获异常_php485错误处理

QQ客服