2013-12-07 52 views
0

我在處理後臺進程時遇到了麻煩。我將這些pid存儲在一個數組中,然後當我調用'jobs'時,它應該遍歷數組,如果數字不爲0,並且它仍然存在,它會打印,如果沒有,它應該刪除PID。我得到非常不一致的結果。有人可以看到我做錯了什麼嗎?感謝 結果:如何存儲進程ID並殺死它們?

ben$ sleep 5 & 
Ben: started job 30290 
ben$ jobs 
30290 
ben$ jobs 
30290 
ben$ jobs 
30290 
ben$ kill 30290 
ben$ kill 30290: No such process 
jobs 
ben$ jobs 
ben$ sleep 4s & 
Ben: started job 30547 
ben$ jobs 
30547 
ben$ kill 30547 
ben$ jobs 
30547 
ben$ jobs 
30547 
ben$ 

第一個文件是主要的,而第二個是我的職能文件

/* 
    Shell 1 
*/ 

#include <stdio.h> 
#include <stdlib.h> 
#include <string.h> 
#include <errno.h> 
#include <unistd.h> 
#include <sys/wait.h> 
#include <stdbool.h> 
#define DELIMS " \t\r\n" 
int main (int argc, char *argv[]) { 
    int ids[10]; 

    initializeArray(ids); 

    while (1) { 
     char line[100]; 
     char *temp; 
     char *split[15]; 
     pid_t pid; 
     bool background=false; 

    printf("ben$ "); 

    //if cmd-d, exit shell 
    if(!fgets(line, sizeof(line), stdin)) { 
     printf("\n"); 
     break; 
    } 

     line[strlen(line)-1]='\0'; 
     temp=strtok(line," "); 


     if (strcmp(temp, "exit") == 0) { 
     break; 
     } 
     else if (strcmp(temp, "cd") == 0) { 
     char *arg = strtok(0, DELIMS); 

     if (!arg) { 
      fprintf(stderr, "cd missing argument.\n"); 
     } 
     else { 
      chdir(arg); 
     } 

     } 
     else if (strcmp(temp, "jobs") == 0) { 
     printJobs(ids); 
     continue; 
     } 

     else { 
     int i=0; 
     while(temp != NULL) { 
     split[i]=temp; 
     temp=strtok(NULL," "); 
     i++; 
     } 
     split[i] = NULL; 
     if(strcmp(split[i-1], "&")==0) { 
     // printf("should do in background"); 
     split[i-1]='\0'; 
     background=true; 
     } 

     char *args[i]; 
     int j; 
     for(j=0; j < i; j++){ 
     args[j]=split[j]; 
     } 


     pid=fork(); 

     if(pid==0) { 
     if(background) { 
        fclose(stdin); 
        fopen("/dev/null","r"); 
     } 
     execvp(args[0], args); 
     printf("This should not happen\n"); 
     } 
     else { 
     if(!background) { 
      // printf("Not in background\n"); 
      wait(&pid); 
     } 
     else { 
      printf("Ben: started job %d\n",pid); 
      insertPid(ids,pid); 
      background=false; 
     } 
    } 
    } 

    } 
    return 0; 
} 


#include <stdio.h> 
#include <stdlib.h> 
#include <string.h> 


void initializeArray(int ids[]); 
void insertPid(int ids[],int pid); 
void removePid(int ids[],int pid); 
void printJobs(int ids[]); 


void initializeArray(int ids[]) { 
    int i; 
    for(i=0;i<10;i++) { 
     ids[i]=0; 
    } 

} 

void insertPid(int ids[],int pid) { 
    // printf("inserting\n"); 
    int i=0; 


    while(i<10) { 
     if(ids[i]==0) { 
      ids[i]=pid; 
      return; 
     } 
     i++; 
    } 
    printf("insert into list error"); 
    return; 
} 
void removePid(int ids[],int index) { 
    // printf("removing %d\n",pid); 
     ids[index]=0; 
     return; 
} 
void printJobs(int ids[]) { 
    // printf("printing\n"); 
    int i; 
    int pid; 
    for(i=0;i<10;i++) { 

     if(ids[i] != 0) { 
      // printf("job value %d",ids[i]); 
      pid=ids[i]; 
      if(kill(pid,0) == 0) { 
       printf("%d\n",pid); 
      } 
      else{ 
       removePid(ids,i); 
      } 
     } 
    } 
} 

回答

0

你怎麼知道,如果你的(背景)的工作已經去世了?你wait在你的前臺工作 - 所以你知道他們什麼時候完成 - 但你不處理後臺工作。無論你是否殺死了孩子,或者他們最終都會自行失效(例如,例子中的睡眠耗盡),你將不會知道它。這也是您在輸出列表中看到的內容。

解決此問題的一種方法是在SIGCHLD和waitpid死亡時設置信號處理程序。然後更新你的作業數組。或者其中的一些變化。