问题描述
我有一个多线程项目。我想将 cpu 关联性设置为特定线程。我正在 Windows 上工作并使用 MinGW 编译器。但是当我初始化类型 cpu_set_t 时,出现错误“类型 cpu_set_t 未在此范围内声明”。我不明白原因。我期待你的帮助!谢谢!
#define _GNU_SOURCE
#include <iostream>
#include <pthread.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sched.h>
int main()
{
int core_id = 1;
cpu_set_t cpuset; // error here
cpu_ZERO(&cpuset);
cpu_SET(core_id,&cpuset);
pthread_t current_thread = pthread_self();
pthread_setaffinity_np(current_thread,sizeof(cpu_set_t),&cpuset);
}
解决方法
Windows 本身不支持 pthreads 标准。 MinGW 项目使用 winpthreads 库作为原生 Windows 线程的包装器。
'_np' 后缀 - 表示不可移植。 pthreads(posix) 标准确实需要这些 *_np 函数。
当前版本的 winpthreads 没有很多在 Linux 等其他操作系统上可以看到的 *_np 函数。
您可以编写自己的函数来更改 Windows 线程关联并从您的线程调用它。 类似的东西:
#include <windows.h>
// Bind thread to CPUs
// Return previous mask value on success,0 on failure
DWORD_PTR BindThreadToCPU(
DWORD mask // 1 - bind to cpu 0
// 4 - bind to cpu 2
// 15 - bind to cpu 0,1,2,3
)
{
HANDLE th = GetCurrentThread();
DWORD_PTR prev_mask = SetThreadAffinityMask(th,mask);
return prev_mask;
}
注意:线程关联掩码必须是进程关联掩码的子集。
还有来自 Microsoft documentation 的关于 SetThreadAffinityMask()
及其调度程序的有趣引用:
线程关联强制线程在特定的处理器子集上运行。 通常应该避免设置线程关联,因为它会干扰 调度程序具有跨处理器有效调度线程的能力。 这会降低并行处理产生的性能提升。 线程亲和性的适当使用是测试每个处理器。
在现代版本的 Windows 中,SetThreadIdealProcessor()
函数对调度程序更友好,并允许您为线程设置首选 CPU(但没有任何严格的亲和性保证)。