2012-06-03 65 views
2

試圖編寫一個處理內部和外部命令的shell。我可以一次獲得內部命令和一個外部命令。使用execv和輸入重定向的C++流水線命令

我的問題是如何獲得這樣的命令來運行:「ls -l命令| grep的LIB |廁所-l」

我用叉子(),並通過execv通過外部命令()在一個char * []中。

有關如何工作的任何想法?我假設使用pipe()或其他東西,但我不確定。

問題的第二部分:如何處理I/O重定向?任何人都可以指給我一個有用的地方嗎?

編輯 到目前爲止,@Alex W是我的英雄。然而,由於我是pipe()和dup2()命令的新手,我對每個調用和變量都有些猶豫。

下面的代碼我有一個處理一個單一的外部命令(例如=「ls -l命令-A」):

pid_t pid; 
    pid = fork(); 
    if (pid < 0) 
    { 
     cout << "Fork failed." << endl; 
    } 
    else if (pid == 0) 
    { 
     execvp(exec_args[0], exec_args); //exec_args is a char*[] where 
     _exit (EXIT_FAILURE);    //exec_args[0] contains "/bin/ls" 
    }           //[1]="ls" and [2]="-l" [3]="-a" 
    else 
    { 
     int status; 
     waitpid(pid, &status, 0); 
    } 
    break; 
+0

你建議的方式正是我如何做到的。 (順便說一下,您可能會將您的問題標記爲「posix」,因爲它不適用於Windows程序員 - 即使您必須刪除其中一個其他標記才能執行此操作。) – thb

+0

我在標記中添加了posix。基本上我正在尋找一些有關「pipe()」如何工作的見解 – twsmale

+0

我明白了。試試'man 7 pipe'。 (您可能知道手冊的第7部分是系統概念記錄的地方。) – thb

回答

2

你使用pipe和POSIX系統的管道工程以類似的方式來一份文件。您可以寫入管道並從管道中讀取,但如果管道中沒有任何內容會阻塞。查找有關Unix系統調用信息的最佳位置是系統調用的手冊頁。在Unix系統上,您可以在終端中輸入man pipe。如果你沒有訪問Unix終端,那麼只需google「man pipe」。關於手冊頁的好處是他們告訴你爲給定的系統調用包含哪些庫。確保你記得在使用任何exec類型系統調用時,你正在將一個全新的進程加載到該內存中,並且正在執行的進程將在其軌道中停止。

要使用它做這樣的事情:

int main() 
{ 
    int id[2]; 
    int fd[2]; 
    int fd2[2]; 
    int fd3[2]; 
    FILE file; 
    int status; 
    int sz = 0; 
    char buff[1000]; 
    char buff2[1000]; 
    string launch[2]; 
    FILE *fp; 

    launch[0] = "./anotherProgramToExecute"; 
    launch[1] = "./yetAnotherProgram"; 

    pipe(fd); 
    pipe(fd2); 
    pipe(fd3); 

    for(int i = 0; i < 2; i++) 
    { 
     id[i] = fork(); 

     if (id[i] == -1) /* an error occurred */ 
     { 
      perror("Fork failed.\n"); 
     } 
     else if (id[i] == 0) /* this is the child process currently acting */ 
     { 
      if(i == 0) 
      { 
       dup2(fd[1],1); 
      } 
      else if(i == 1) 
      { 
       dup2(fd2[0],0); 
       dup2(fd3[1],1); 
      } 
      execlp(launch[i],launch[i], NULL); 
     } 
     else /* this is the parent process currently acting */ 
     { 
      sz = read(fd[0], buff, 1000); 
      buff[sz] = '\0'; 
      printf("buff = %s\n",buff); 

      close(fd[0]); 

      write(fd2[1],buff, 1000); 

      read(fd3[0],buff2,1000); 

      fp = fopen("bin.txt","w"); 
      if(fp == NULL) 
       printf("Cannot open file.\n"); 
      else 
      { 
       fprintf(fp,buff2); 
       fclose(fp); 
      } 

      //printf("Creation of Child Process #%d succeeded!\n",id[i]); 
      while(waitpid(id[i], &status, WNOHANG) == 0) 
       sleep(0.3); 
      if (WIFEXITED(status)) 
      { 
       // successfully terminated children 
      } 
      else 
      { 
       perror("Child has not terminated correctly.\n"); 
      } 
     } 
    } 
} 
+0

非常有幫助,但是你能否偶然評論一些變量,以便我知道每個人都在做什麼?就像我說的,我是一個管道()noob – twsmale

+0

當你調用pipe(fd)時,它用兩個文件描述符填充int fd [2]數組。在索引0處,它將放置讀取文件描述符,並且在索引1處(即fd [1]),它將放置寫入文件描述符。您可以使用它們來讀取和寫入管道,就像它是一個常規文件一樣。 –

0

最好的在線教程關於管道和這種是一個過時的歌曲,但糖果我所看到的:http://beej.us/guide/bgipc/。你可以瀏覽它。有人懷疑你會喜歡它。