根據PID獲取進程名&根據進程名獲取PID


Liunx中 通過進程名查找進程PID可以通過 pidof [進程名] 來查找。反過來 ,相同通過PID查找進程名則沒有相關命令。
在linux根目錄中,有一個/proc的VFS(虛擬文件系統),系統當前運行的所有進程都對應於該目錄下的一個 以進程PID命名的文件夾 ,其中存放進程運行的N多信息。其中有一個status文件,cat顯示該文件, 第一行的Name 即為進程名。

 

打開stardict程序,進程名為stardict;

shell中分別根據Pid獲取進程名、根據進程名獲取Pid

1)查找stardict的pid:pidof stardict

2)根據1)的pid查找進程名: grep "Name:" /proc/5884/status

 

應用:kill一個進程需要指定該進程的pid,所以我們需要先根據進程名找到pid,然后再kill;

   killall命令則只需要給定進程名即可,應該是封裝了這個過程。

 

 

C程序中實現上述過程

 1 #include <sys/types.h>
 2 #include <dirent.h>
 3 #include <stdio.h>
 4 #include <string.h>
 5  
 6 #define BUF_SIZE 1024
 7 
 8 void getPidByName(pid_t *pid, char *task_name)
 9 {
10     DIR *dir;
11     struct dirent *ptr;
12     FILE *fp;
13     char filepath[50];
14     char cur_task_name[50];
15     char buf[BUF_SIZE];
16 
17     dir = opendir("/proc"); 
18     if (NULL != dir)
19     {
20         while ((ptr = readdir(dir)) != NULL) //循環讀取/proc下的每一個文件/文件夾
21         {
22             //如果讀取到的是"."或者".."則跳過,讀取到的不是文件夾名字也跳過
23             if ((strcmp(ptr->d_name, ".") == 0) || (strcmp(ptr->d_name, "..") == 0))
24                 continue;
25             if (DT_DIR != ptr->d_type)
26                 continue;
27            
28             sprintf(filepath, "/proc/%s/status", ptr->d_name);//生成要讀取的文件的路徑
29             fp = fopen(filepath, "r");
30             if (NULL != fp)
31             {
32                 if( fgets(buf, BUF_SIZE-1, fp)== NULL ){
33                     fclose(fp);
34                     continue;
35                 }
36                 sscanf(buf, "%*s %s", cur_task_name);
37         
38                 //如果文件內容滿足要求則打印路徑的名字(即進程的PID)
39                 if (!strcmp(task_name, cur_task_name)){
40                     sscanf(ptr->d_name, "%d", pid);
41                 }
42                 fclose(fp);
43             }
44         }
45         closedir(dir);
46     }
47 }
48 
49 void getNameByPid(pid_t pid, char *task_name) {
50     char proc_pid_path[BUF_SIZE];
51     char buf[BUF_SIZE];
52 
53     sprintf(proc_pid_path, "/proc/%d/status", pid);
54     FILE* fp = fopen(proc_pid_path, "r");
55     if(NULL != fp){
56         if( fgets(buf, BUF_SIZE-1, fp)== NULL ){
57             fclose(fp);
58         }
59         fclose(fp);
60         sscanf(buf, "%*s %s", task_name);
61     }
62 }
63 
64 void main(int argc, char** argv)
65 {
66     char task_name[50];
67     pid_t pid = getpid();
68 
69     printf("pid of this process:%d\n", pid);
70     getNameByPid(pid, task_name);
71     
72     /*
73     strcpy(task_name, argv[0]+2);
74     printf("task name is %s\n", task_name);
75     getPidByName(task_name);
76     */
77     printf("getNameByPid:%s\n", task_name);
78     getPidByName(&pid, task_name);
79     printf("getPidByName:%d\n", pid);
80     sleep(15);
81 }

 

運行結果:

 

進入/proc/4173/status查看文件內容,一切對應。

Name: test
State: S (sleeping)
Tgid: 4173
Pid: 4173
PPid: 2721
TracerPid: 0
Uid: 1000 1000 1000 1000
Gid: 1000 1000 1000 1000
FDSize: 256
Groups: 4 24 27 30 46 109 124 1000
VmPeak: 4300 kB
VmSize: 4296 kB
VmLck: 0 kB
VmPin: 0 kB
VmHWM: 600 kB
VmRSS: 600 kB
VmData: 180 kB
VmStk: 136 kB
VmExe: 4 kB
VmLib: 1884 kB
VmPTE: 32 kB
VmSwap: 0 kB
Threads: 1
SigQ: 0/27055
SigPnd: 0000000000000000
ShdPnd: 0000000000000000
SigBlk: 0000000000000000
SigIgn: 0000000000000000
SigCgt: 0000000000000000
CapInh: 0000000000000000
CapPrm: 0000000000000000
CapEff: 0000000000000000
CapBnd: ffffffffffffffff
Cpus_allowed: f
Cpus_allowed_list: 0-3
Mems_allowed: 00000000,00000001
Mems_allowed_list: 0
voluntary_ctxt_switches: 2
nonvoluntary_ctxt_switches: 9


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM