linux中通过proc获取进程名以及PID

  • 背景
    • 给定某个线程的线程号tid,获取对应的进程名
  • 解决
    • 刚开始以为有难度,需要遍历整个proc/***以及/proc/***/task/***;因为拿到的是tid而非pid,该tid是由某个pid通过pthread_create或fork、clone出来的,只有找到对应的pid,才能从/proc/***/status下获取进程名;
    • 但是linux下,tid和pid就是一个东西,内核中是完全不区分这两个概念(即线程和进程);体现在/proc下就是,不论tid还是pid,都可以在根目录下找到;因此直接读取/proc/tid/status文件即可获取进程名
    • 当然,如果tid是由某个pid创建出来的,在/proc/pid/task是可以找到对应tid的,这也就是pstree工具的实现原理;
    • 从这也可以看出,fork以及pthread_create其实最终到内核中,使用的都是一条创建路径;由此也可以看出,在linux中没有所谓的进程和线程之分,内核中,都是进程,就是一个task_struct结构体,重而进行执行调度
  • 源码
 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);
     }
 }
  • 延伸:通过进程名获取pid
    引用:http://www.voidcn.com/article/p-kszfbyda-ng.html
void getPidByName(pid_t *pid, char *task_name)
 {
     DIR *dir;
     struct dirent *ptr;
     FILE *fp;
     char filepath[50];
     char cur_task_name[50];
     char buf[BUF_SIZE];

     dir = opendir("/proc"); 
     if (NULL != dir)
     {
         while ((ptr = readdir(dir)) != NULL) //循环读取/proc下的每一个文件/文件夹
         {
             //如果读取到的是"."或者".."则跳过,读取到的不是文件夹名字也跳过
             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)){
                     sscanf(ptr->d_name, "%d", pid);
                 }
                 fclose(fp);
             }
         }
         closedir(dir);
     }
 }

你可能感兴趣的:(Linu内核,proc,pid,进程名)