2013-03-06 68 views
1

我正在另一個shell中運行這個基本的shell程序。我無法弄清楚爲什麼在執行「ls」後我的shell不能繼續運行。我沒有退出它,但它回到原來的殼。如果想要使用它,我必須每次運行我的shell程序。我想這是什麼叉()應該做的。我只希望我的shell使用退出命令退出,我用if語句編碼。任何建議將不勝感激。哦,不理會gettoks()解析器函數,我不知道如何使用它輸入,所以我寫了如果其他語句的字符串輸入cmSTR而不是使用gettoks()解析器。主要是因爲我想不出如何將輸入傳遞到它如何讓我的shell在命令後運行

#include <iostream> 
#include <unistd.h> 
#include <sys/types.h> 
#include <errno.h> 
#include <signal.h> 
#include <cstdlib> 
#include <sys/wait.h> 

using namespace std; 
// Initializing counters for trapping 
static int cc_counter = 0; 
static int cz_counter = 0; 
static int cq_counter = 0; 
//Functions for trap signal handler 
void cc_handler(int signo) 
{ 
++cc_counter; 
} 
void cz_handler(int signo) 
{ 
++cz_counter; 
} 
void cq_handler(int signo) 
{ 
++cq_counter; 
} 

//********************************************************* 
// 
// Extern Declarations 
// 
//********************************************************* 
using namespace std; 
extern "C" 
{ 
extern char **gettoks(); 
} 


//********************************************************* 
// 
// Main Function 
// 
//********************************************************* 
int main(int argc, char *argv[]) 
{ 
// local variables 
int ii; 
char **toks; 
int retval; 

// initialize local variables 
ii = 0; 
toks = NULL; 
retval = 0; 
char buf[1000];//Initialize of size for current working directory 
string cmSTR;//String to hold input 
int status;//Initialization of status for fork() 
pid_t pid;//Declaration of pid 

// main (infinite) loop 
while(true) 
{ 
    signal(SIGINT, cc_handler);// Traps Ctrl+C 
    signal(SIGTSTP, cz_handler);// Traps Ctrl+Z 
    signal(SIGQUIT, cq_handler);// Traps Ctrl+\ 
    //prompt and show current working directory 
    cout <<("RS_SHELL:") << getcwd(buf,1000) << "\t"; 
    getline(cin ,cmSTR);//read input from keyboard 
    // if else loop to switch based on command input 
    if(cmSTR == "ls")// if ls, then execute arguement 
    { 
    execl("/bin/ls", "ls", NULL);//System call to execute ls 

    } 
    else if(cmSTR == "exit")//if exit, then execute block of code 
    { 
     cout << "Ctrl C entered: " << ++cc_counter << "times"<< endl; 
     cout << "Ctrl Z entered: " << ++cz_counter << "times"<< endl; 
     cout << "Ctrl Back Slash entered: " << ++cq_counter << "times"<< endl; 
     exit(1); 
     } 
    else if(cmSTR == "guish")// if guish, execute guish shell 
    { 
     execvp("guish", NULL); 
     } 
     //if input is not any of previous commands then fork() 
    else if(cmSTR != "ls" && cmSTR != "exit" && cmSTR != "guish" && cmSTR != "\n") 
    { 
     pid = fork(); 
     if (pid < 0)//Loop to fork parent and child process 
     { 
      fprintf(stderr, "Fork Failed"); 
      exit(-1); 
      } 
      else if (pid == 0)//Child process 
     { 
      execvp("guish", NULL);//system call to execute guish shell 

      } 
      else //Parent process 
      { 
       waitpid(-1, &status,0); 
       exit(0); 
       } 
       } 




    // get arguments 
    toks = gettoks(); 

    if(toks[0] != NULL) 
{ 
    // simple loop to echo all arguments 
    for(ii=0; toks[ii] != NULL; ii++) 
    { 
     cout << "Argument " << ii << ": " << toks[ii] << endl; 
    } 

    if(!strcmp(toks[0], "exit")) 
    break; 
} 
    } 

    // return to calling environment 
    return(retval); 
    } 

回答

0

當你懷疑,execl及其相關功能疊加一個新的進程當前進程。因此,在啓動lsexecl調用之後,程序將不再存在以繼續運行。

如果你希望你的shell程序運行後ls周圍停留,你需要fork()通話execl("/bin/ls", "ls", NULL);之前。另外,如果你想從ls的輸出出現在你的shell的同一個控制檯中,我想你可能打算這樣做,你需要將輸出從ls返回到你的shell,然後將輸出寫到你的外殼的控制檯。例如參見Writing my own shell… stuck on pipes?

+0

以外的if語句? – user2125805 2013-03-06 22:31:56

+0

是的,我會傾向於'if(if)'之前的'fork()',而不是分別爲每個'if'個案分叉 - DRY(不要重複自己)是一個很好的座右銘。 – Simon 2013-03-07 09:39:16