CPU亲合力就是指在Linux系统中能够将一个或多个进程绑定到一个或多个处理器上运行.
一个进程的CPU亲合力掩码决定了该进程将在哪个或哪几个CPU上运行.在一个多处理器系统中,设置CPU亲合力的掩码可能会获得更好的性能.
一个CPU的亲合力掩码用一个cpu_set_t结构体来表示一个CPU集合,下面的几个宏分别对这个掩码集进行操作:
·CPU_ZERO() 清空一个集合
·CPU_SET()与CPU_CLR()分别对将一个给定的CPU号加到一个集合或者从一个集合中去掉.
·CPU_ISSET()检查一个CPU号是否在这个集合中.

下面两个函数就是用来设置获取线程CPU亲和力状态: 
    ·sched_setaffinity(pid_t
pid, unsigned int cpusetsize, cpu_set_t *mask) 
      该函数设置进程为pid的这个进程,让它运行在mask所设定的CPU上.如果pid的值为0,则表示指定的是当前进程,使当前进程运行在mask所设定的那些CPU上.第二个参数cpusetsize是mask所指定的数的长度.通常设定为sizeof(cpu_set_t).如果当前pid所指定的进程此时没有运行在mask所指定的任意一个CPU上,则该指定的进程会从其它CPU上迁移到mask的指定的一个CPU上运行. 
    ·sched_getaffinity(pid_t
pid, unsigned int cpusetsize, cpu_set_t *mask) 
      该函数获得pid所指示的进程的CPU位掩码,并将该掩码返回到mask所指向的结构中.即获得指定pid当前可以运行在哪些CPU上.同样,如果pid的值为0.也表示的是当前进程

  1. cpu_set_t的定义
  2. # define __CPU_SETSIZE 1024
  3. # define __NCPUBITS (8 * sizeof (__cpu_mask))
  4. typedef unsigned long int __cpu_mask;
  5. # define __CPUELT(cpu) ((cpu) / __NCPUBITS)
  6. # define __CPUMASK(cpu) ((__cpu_mask) 1 << ((cpu) % __NCPUBITS))
  7. typedef struct
  8. {
  9. __cpu_mask __bits[__CPU_SETSIZE / __NCPUBITS];
  10. } cpu_set_t;
  11. # define __CPU_ZERO(cpusetp) \
  12. do { \
  13. unsigned int __i; \
  14. cpu_set_t *__arr = (cpusetp); \
  15. for (__i = 0; __i < sizeof (cpu_set_t) / sizeof (__cpu_mask); ++__i) \
  16. __arr->__bits[__i] = 0; \
  17. } while (0)
  18. # define __CPU_SET(cpu, cpusetp) \
  19. ((cpusetp)->__bits[__CPUELT (cpu)] |= __CPUMASK (cpu))
  20. # define __CPU_CLR(cpu, cpusetp) \
  21. ((cpusetp)->__bits[__CPUELT (cpu)] &= ~__CPUMASK (cpu))
  22. # define __CPU_ISSET(cpu, cpusetp) \
  23. (((cpusetp)->__bits[__CPUELT (cpu)] & __CPUMASK (cpu)) != 0)

测试代码:

[cpp] view plain copy

  1. #include<stdlib.h>
  2. #include<stdio.h>
  3. #include<sys/types.h>
  4. #include<sys/sysinfo.h>
  5. #include<unistd.h>
  6. #define __USE_GNU
  7. #include<sched.h>
  8. #include<ctype.h>
  9. #include<string.h>
  10. #include<pthread.h>
  11. #define THREAD_MAX_NUM 100  //1个CPU内的最多进程数
  12. int num=0;  //cpu中核数
  13. void* threadFun(void* arg)  //arg  传递线程标号(自己定义)
  14. {
  15. cpu_set_t mask;  //CPU核的集合
  16. cpu_set_t get;   //获取在集合中的CPU
  17. int *a = (int *)arg;
  18. printf("the a is:%d\n",*a);  //显示是第几个线程
  19. CPU_ZERO(&mask);    //置空
  20. CPU_SET(*a,&mask);   //设置亲和力值
  21. if (sched_setaffinity(0, sizeof(mask), &mask) == -1)//设置线程CPU亲和力
  22. {
  23. printf("warning: could not set CPU affinity, continuing...\n");
  24. }
  25. while (1)
  26. {
  27. CPU_ZERO(&get);
  28. if (sched_getaffinity(0, sizeof(get), &get) == -1)//获取线程CPU亲和力
  29. {
  30. printf("warning: cound not get thread affinity, continuing...\n");
  31. }
  32. int i;
  33. for (i = 0; i < num; i++)
  34. {
  35. if (CPU_ISSET(i, &get))//判断线程与哪个CPU有亲和力
  36. {
  37. printf("this thread %d is running processor : %d\n", i,i);
  38. }
  39. }
  40. }
  41. return NULL;
  42. }
  43. int main(int argc, char* argv[])
  44. {
  45. num = sysconf(_SC_NPROCESSORS_CONF);  //获取核数
  46. pthread_t thread[THREAD_MAX_NUM];
  47. printf("system has %i processor(s). \n", num);
  48. int tid[THREAD_MAX_NUM];
  49. int i;
  50. for(i=0;i<num;i++)
  51. {
  52. tid[i] = i;  //每个线程必须有个tid[i]
  53. pthread_create(&thread[0],NULL,threadFun,(void*)&tid[i]);
  54. }
  55. for(i=0; i< num; i++)
  56. {
  57. pthread_join(thread[i],NULL);//等待所有的线程结束,线程为死循环所以CTRL+C结束
  58. }
  59. return 0;
  60. }

编译命令:gcc bind.c -o bind -lpthread

执行:./bind

输出结果:略

特别注意:

#define __USE_GNU不要写成#define _USE_GNU

#include<pthread.h>必须写在#define __USE_GNU之后,否则编译会报错

查看你的线程情况可以在执行时在另一个窗口使用top -H来查看线程的情况,查看各个核上的情况请使用top命令然后按数字“1”来查看。

原文:https://blog.csdn.net/lanyzh0909/article/details/50404664

线程绑定CPU核-sched_setaffinity的更多相关文章

  1. Ubuntu系统进程绑定CPU核

    Ubuntu系统进程绑定CPU核 作者:chszs.版权全部,未经允许,不得转载. 博主主页:http://blog.csdn.net/chszs 本文讲述如何在Ubuntu系统中,把指定的进程绑定到 ...

  2. Linux编程之《进程/线程绑定CPU》

    Intro----- 通常我们在编写服务器代码时,可以通过将当前进程绑定到固定的CPU核心或者线程绑定到固定的CPU核心来提高系统调度程序的效率来提高程序执行的效率,下面将完整代码贴上. /***** ...

  3. taskset -pc PID 查看线程占用cpu核

    taskset -pc  PID 可以用于 查看 当前线程 对应绑定的 在 哪个核上面. 这个 可以用于 程序优化, 查看 哪个线程占用的 cpu 比重比较高 首先 可以通过  top  -H   - ...

  4. linux线程绑定cpu

    函数介绍 #define __USE_GNU #include <sched.h> void CPU_ZERO(cpu_set_t *set); void CPU_SET(int cpu, ...

  5. 为线程绑定CPU

    // learn gcc atomic variable #define _GNU_SOURCE #include <stdio.h> #include <stdlib.h> ...

  6. 线程绑定cpu

    #include <stdio.h> #include <pthread.h> #include <sys/sysinfo.h> #include <unis ...

  7. Windows10 临时将线程绑定至指定CPU的方法

    本文首发:https://www.somata.work/2019/WindowsThreadBind.html 将线程绑定至指定CPU,这个应该时很多管理员需要了解认知的操作了吧,这样可以在一定程度 ...

  8. NGINX源代码剖析 之 CPU绑定(CPU亲和性)

    作者:邹祁峰 邮箱:Qifeng.zou.job@gmail.com 博客:http://blog.csdn.net/qifengzou 日期:2014.06.12 18:44 转载请注明来自&quo ...

  9. linux下将不同线程绑定到不同core和cpu上——pthread_setaffinity_np

    =============================================================== linux下的单进程多线程的程序,要实现每个线程平均分配到多核cpu,主 ...

随机推荐

  1. ubuntu 安装 rpm 软件包

    1.首先安装alien和fakeroot这两个软件,alien可以将rpm转换为deb包.命令sudo apt-get install alien fakeroot 2.使用alien将rpm包转为d ...

  2. HDU 2112 HDU Today(STL MAP + Djistra)

    题目链接:HDU Today 立即集训要開始,抓紧时间练练手,最短路的基础题,第一次用STL的map 题目非常水,可是错了N遍.手贱了.本题不优点理的就是把地名转化为数字 #include <i ...

  3. TP 上传excel

    <?php class ExcelAction extends Action{ public function read($filename,$encode='utf-8'){ vendor(' ...

  4. 018 nginx与第三模块整合[一致性哈希模块整合]

    nginx第三方模块官网:http://wiki.nginx.org/HttpUpstreamConsistentHash nginx第三方模块下载地址:https://github.com/repl ...

  5. rpm包查看和解压(转)

    From:http://www.51testing.com/html/57/28557-205195.html 查看rpm包内容: rpm -qpl *.rpm 解压rpm包: rpm2cpio *. ...

  6. 【Atheros】禁用CSMA之后pktgen发包一分钟后无法发送的问题

    无线网络中各个节点不断地广播信标帧,收到某节点的信标帧之后才知道这个节点存在,知道它的网络配置是怎么样的,才能知道应该怎么和它通信. 那么问题来了,禁用了CSMA之后,发送节点全力发送,那么它会永远占 ...

  7. java 给多人发送、抄送

    关键技术: 1.MimeMessage的setRecipients方法设置邮件的收件人,其中Message.RecipientType.TO常量表示收件人类型是邮件接收者,Message.Recipi ...

  8. python 基础 5.2 类的继承

    一. 类的继承 继承,顾名思议就知道是它的意思,举个例子说明,你现在有一个现有的A类,现在需要写一个B类,但是B类是A类的特殊版,我们就可以使用继承,B类继承A类时,B类会自动获得A类的所有属性和方法 ...

  9. EasyPlayerPro windows播放器本地配置文件配置方法介绍

    需求背景 应EasyPlayerPro某客户需求,在EasyPlayerPro启动时,自动播放指定的url源, 不需要每次都去手动填写, 且实现自动播放,不需要手动的单击播放按钮: 为响应该需求,特增 ...

  10. API的理解和使用——有序集合

    有序集合常用的命令 命令 功能 zadd key score member [score member ... ] 添加元素 zcard key 计算成员个数 zscore key member 计算 ...