linux下把程式繫結到特定cpu核上執行
導讀 | 現在大家使用的基本上都是多核cpu,一般是4核的。平時應用程式在執行時都是由作業系統管理的。作業系統對應用程式進行排程,使其在不同的核上輪番執行。 |
對於普通的應用,作業系統的預設排程機制是沒有問題的。但是,當某個程式需要較高的執行效率時,就有必要考慮將其繫結到單獨的核上執行,以減小由於在不同的核上排程造成的開銷。
把某個程式/執行緒繫結到特定的cpu核上後,該程式就會一直在此核上執行,不會再被作業系統排程到其他核上。但繫結的這個核上還是可能會被排程執行其他應用程式的。
目前windows和 都支援對多核cpu進行排程管理。
軟體開發在多核環境下的核心是多執行緒開發。這個多執行緒不僅代表了軟體實現上多執行緒,要求在硬體上也採用多執行緒技術。
多核作業系統的關注點在於程式的分配和排程。程式的分配將程式分配到合理的物理核上,因為不同的核在共享性和歷史執行情況都是不同的。有的物理核能夠共享二級cache,而有的卻是獨立的。如果將有資料共享的程式分配給有共享二級cache的核上,將大大提升效能;反之,就有可能影響效能。
程式排程會涉及實時性、負載均衡等問題,目前研究的熱點問題主要集中在以下方面:
- 程式的並行開發設計
- 多程式的時間相關性
- 任務的分配和排程
- 快取的錯誤共享
- 一致性訪問問題
- 程式間通訊
- 多處理器核內部資源競爭
多程式和多執行緒在cpu核上執行時情況如下:
每個 CPU 核執行一個程式的時候,由於每個程式的資源都獨立,所以 CPU 核心之間切換的時候無需考慮上下文
每個 CPU 核執行一個執行緒的時候,有時執行緒之間需要共享資源,所以這些資源必須從 CPU 的一個核心被複制到另外一個核心,這會造成額外的開銷
使用 cat /proc/cpuinfo檢視cpu資訊,如下兩個資訊:
processor,指明第幾個cpu處理器
cpu cores,指明每個處理器的核心數
也可以使用系統呼叫sysconf獲取cpu核心數:
#include <unistd.h> int sysconf(_SC_NPROCESSORS_CONF);/* 返回系統可以使用的核數,但是其值會包括系統中禁用的核的數目,因 此該值並不代表當前系統中可用的核數 */ int sysconf(_SC_NPROCESSORS_ONLN);/* 返回值真正的代表了系統當前可用的核數 */ /* 以下兩個函式與上述類似 */ #include <sys/sysinfo.h> int get_nprocs_conf (void);/* 可用核數 */ int get_nprocs (void);/* 真正的反映了當前可用核數 */
我使用的是虛擬機器,有2個處理器,每個處理器只有一個核,等同於一個處理器兩個核心。
獲取程式pid
-> % ps PID TTY TIME CMD 2683 pts/1 00:00:00 zsh 2726 pts/1 00:00:00 dgram_servr 2930 pts/1 00:00:00 ps
檢視程式當前執行在哪個cpu上
-> % taskset -p 2726 pid 2726's current affinity mask: 3
顯示的十進位制數字3轉換為2進製為最低兩個是1,每個1對應一個cpu,所以程式執行在2個cpu上。
指定程式執行在cpu1上
-> % taskset -pc 1 2726 pid 2726's current affinity list: 0,1 pid 2726's new affinity list: 1
注意,cpu的標號是從0開始的,所以cpu1表示第二個cpu(第一個cpu的標號是0)。
至此,就把應用程式繫結到了cpu1上執行,檢視如下:
-> % taskset -p 2726 pid 2726's current affinity mask: 2
啟動程式時繫結cpu
#啟動時繫結到第二個cpu -> % taskset -c 1 ./dgram_servr& [1] 3011 #檢視確認繫結情況 -> % taskset -p 3011 pid 3011's current affinity mask: 2
sched_setaffinity可以將某個程式繫結到一個特定的CPU。
#define _GNU_SOURCE /* See feature_test_macros(7) */ #include <sched.h> /* 設定程式號為pid的程式執行在mask所設定的CPU上 * 第二個引數cpusetsize是mask所指定的數的長度 * 通常設定為sizeof(cpu_set_t) * 如果pid的值為0,則表示指定的是當前程式 */ int sched_setaffinity(pid_t pid, size_t cpusetsize, cpu_set_t *mask); int sched_getaffinity(pid_t pid, size_t cpusetsize, cpu_set_t *mask);/* 獲得pid所指示的程式的CPU位掩碼,並將該掩碼返回到mask所指向的結構中 */
例項
#include<stdlib.h> #include<stdio.h> #include<sys/types.h> #include<sys/sysinfo.h> #include<unistd.h> #define __USE_GNU #include<sched.h> #include<ctype.h> #include<string.h> #include<pthread.h> #define THREAD_MAX_NUM 200 //1個CPU內的最多程式數 int num=0; //cpu中核數 void* threadFun(void* arg) //arg 傳遞執行緒標號(自己定義) { cpu_set_t mask; //CPU核的集合 cpu_set_t get; //獲取在集合中的CPU int *a = (int *)arg; int i; printf("the thread is:%d\n",*a); //顯示是第幾個執行緒 CPU_ZERO(&mask); //置空 CPU_SET(*a,&mask); //設定親和力值 if (sched_setaffinity(0, sizeof(mask), &mask) == -1)//設定執行緒CPU親和力 { printf("warning: could not set CPU affinity, continuing...\n"); } CPU_ZERO(&get); if (sched_getaffinity(0, sizeof(get), &get) == -1)//獲取執行緒CPU親和力 { printf("warning: cound not get thread affinity, continuing...\n"); } for (i = 0; i < num; i++) { if (CPU_ISSET(i, &get))//判斷執行緒與哪個CPU有親和力 { printf("this thread %d is running processor : %d\n", i,i); } } return NULL; } int main(int argc, char* argv[]) { int tid[THREAD_MAX_NUM]; int i; pthread_t thread[THREAD_MAX_NUM]; num = sysconf(_SC_NPROCESSORS_CONF); //獲取核數 if (num > THREAD_MAX_NUM) { printf("num of cores[%d] is bigger than THREAD_MAX_NUM[%d]!\n", num, THREAD_MAX_NUM); return -1; } printf("system has %i processor(s). \n", num); for(i=0;i<num;i++) { tid[i] = i; //每個執行緒必須有個tid[i] pthread_create(&thread[i],NULL,threadFun,(void*)&tid[i]); } for(i=0; i< num; i++) { pthread_join(thread[i],NULL);//等待所有的執行緒結束,執行緒為死迴圈所以CTRL+C結束 } return 0; }
執行結果
-> % ./a.out system has 2 processor(s). the thread is:0 the thread is:1 this thread 0 is running processor : 0 this thread 1 is running processor : 1
繫結執行緒到cpu核上使用pthread_setaffinity_np函式,其原型定義如下:
#define _GNU_SOURCE /* See feature_test_macros(7) */ #include <pthread.h> int pthread_setaffinity_np(pthread_t thread, size_t cpusetsize, const cpu_set_t *cpuset); int pthread_getaffinity_np(pthread_t thread, size_t cpusetsize, cpu_set_t *cpuset); Compile and link with -pthread.
各引數的意義與sched_setaffinity相似。
例項
#define _GNU_SOURCE #include <pthread.h> #include <stdio.h> #include <stdlib.h> #include <errno.h> #define handle_error_en(en, msg) \ do { errno = en; perror(msg); exit(EXIT_FAILURE); } while (0) int main(int argc, char *argv[]) { int s, j; cpu_set_t cpuset; pthread_t thread; thread = pthread_self(); /* Set affinity mask to include CPUs 0 to 7 */ CPU_ZERO(&cpuset); for (j = 0; j < 8; j++) CPU_SET(j, &cpuset); s = pthread_setaffinity_np(thread, sizeof(cpu_set_t), &cpuset); if (s != 0) handle_error_en(s, "pthread_setaffinity_np"); /* Check the actual affinity mask assigned to the thread */ s = pthread_getaffinity_np(thread, sizeof(cpu_set_t), &cpuset); if (s != 0) handle_error_en(s, "pthread_getaffinity_np"); printf("Set returned by pthread_getaffinity_np() contained:\n"); for (j = 0; j < CPU_SETSIZE; j++) if (CPU_ISSET(j, &cpuset)) printf(" CPU %d\n", j); exit(EXIT_SUCCESS); }
執行結果
-> % ./a.out Set returned by pthread_getaffinity_np() contained: CPU 0 CPU 1
可以使用多種方法把程式/執行緒指定到特定的cpu核上執行。
本文原創地址:
來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/69955379/viewspace-2839059/,如需轉載,請註明出處,否則將追究法律責任。
相關文章
- Linux 中執行緒與 CPU 核的繫結Linux執行緒
- Windows下繫結執行緒到指定的CPU核心Windows執行緒
- 執行緒繫結cpu核心的程式碼研究執行緒
- cpu、核與執行緒執行緒
- Linux下區分物理CPU、邏輯CPU和CPU核數Linux
- 4核8執行緒和6核6執行緒的CPU哪個好?電腦CPU核數多和執行緒多的區別執行緒
- Linux下判斷cpu物理個數,幾核,是超執行緒還是多核心Linux執行緒
- 檢查特定埠上正在執行的程式名(轉)
- linux 下檢視物理CPU,邏輯CPU,CPU core,超執行緒Linux執行緒
- 對多執行緒程式,單核cpu與多核cpu如何工作相關的探討執行緒單核
- 1、多執行緒同步——CPU、core核、執行緒、記憶體執行緒記憶體
- Linux 中如何設定每個特定的時間執行特定的程式Linux
- 執行計劃繫結
- java高階用法之:繫結CPU的執行緒Thread-AffinityJava執行緒thread
- Linux及AIX下如何檢視物理CPU, 邏輯CPU及核數LinuxAI
- 上傳執行sql檔案到linuxSQLLinux
- 根據CPU核數合理設定執行緒池大小執行緒
- netty系列之:在netty中實現執行緒和CPU繫結Netty執行緒
- 如何 SSH 登入到 Linux 上的特定目錄Linux
- Linux--程序繫結NUMA節點或cpu核心Linux
- 聊聊CPU的發展歷程之單核、多核、超執行緒單核執行緒
- react中在函式繫結時會馬上執行.沒有加括號不會馬上執行React函式
- sqm執行計劃的繫結
- Oracle-繫結執行計劃Oracle
- 如何檢視CPU核數和執行緒數?CPU的核心數、執行緒數的關係和區別執行緒
- linux下udev重新繫結磁碟Linuxdev
- 檢視伺服器CPU的個數、CPU的核數、多核超執行緒數伺服器執行緒
- 程式中使用繫結變數,執行計劃不正確變數
- [轉帖]Redis如何繫結CPURedis
- Java執行時繫結探討薦Java
- WIN32下執行緒和視窗的資料繫結Win32執行緒
- 把 Linux 移植到蘋果 M1 Mac 上Linux蘋果Mac
- 請教一個程式執行時,從磁碟到記憶體到CPU的過程?記憶體
- 用pyinstaller打包你的Python程式並繫結CPUPython
- 在 Linux 中查詢 CPU 的核數Linux
- 在LINUX下編寫程式並執行Linux
- iOS多執行緒程式設計總結(上)iOS執行緒程式設計
- Linux 7下使用udev繫結ASM磁碟LinuxdevASM