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程序中實現上述過程
#define BUF_SIZE 1024
void getPidByName(char* task_name)
{
DIR *dir;
struct dirent *ptr;
FILE *fp;
char filepath[50];//大小隨意,能裝下cmdline文件的路徑即可
char cur_task_name[50];//大小隨意,能裝下要識別的命令行文本即可
char buf[BUF_SIZE];
dir = opendir("/proc"); //打開路徑
if (NULL != dir)
{
while ((ptr = readdir(dir)) != NULL) //循環讀取路徑下的每一個文件/文件夾
{
//如果讀取到的是"."或者".."則跳過,讀取到的不是文件夾名字也跳過
if ((strcmp(ptr->d_name, ".") == 0) || (strcmp(ptr->d_name, "..") == 0))
continue;
if (DT_DIR != ptr->d_type)
continue;
sprintf(filepath, "/proc/%s/status", ptr->d_name);//生成要讀取的文件的路徑
fp = fopen(filepath, "r");//打開文件
if (NULL != fp)
{
if( fgets(buf, BUF_SIZE-1, fp)== NULL ){
fclose(fp);
continue;
}
sscanf(buf, "%*s %s", cur_task_name);
//如果文件內容滿足要求則打印路徑的名字(即進程的PID)
if (!strcmp(task_name, cur_task_name))
printf("PID: %s/n", ptr->d_name);
fclose(fp);
}
}
closedir(dir);//關閉路徑
}
}
void getNameByPid(pid_t pid, char *task_name) {
char proc_pid_path[BUF_SIZE];
char buf[BUF_SIZE];
sprintf(proc_pid_path, "/proc/%d/status", pid);
FILE* fp = fopen(proc_pid_path, "r");
if(NULL != fp){
if( fgets(buf, BUF_SIZE-1, fp)== NULL ){
fclose(fp);
}
fclose(fp);
sscanf(buf, "%*s %s", task_name);
}
}
void main(int argc, char** argv)
{
char task_name[50];
pid_t pid = getpid();
printf("pid of this process:%d/n", pid);
getNameByPid(pid, task_name);
/*
strcpy(task_name, argv[0]+2);
printf("task name is %s/n", task_name);
getPidByName(task_name);
*/
printf("task name is %s/n", task_name);
getPidByName(task_name);
sleep(15);
}
運行結果:
進入/proc/9674/status查看文件內容,一切對應。
新聞熱點
疑難解答
圖片精選