簡體   English   中英

如何執行Shell重定向C

[英]How to do shell redirection C

我有一個分詞器,可以接收用戶的輸入。 從那里繼續掃描“>”或“ <”進行潛在的重定向。

for(i=0;i<n;i++){  //I/O redirection
        //printf("extracted token is %s\n",tokens[i]);

        //get the
        if (strcmp(tokens[i],"<")==0) {
            printf("found <  !!!!\n");

            if(tokens[i+1] == NULL){  //if there isn't a input file
                printf("Please have a input file\n");
                break;
            }else if(tokens[i-1] == NULL){ //if there isn't a output file
                printf("Pleae have a output file\n");
                break;
            }
            infile = 1;
            outfile = 1;

            fd=fopen(tokens[i-1],"w");
            fclose(fd);
        }
 }

上面的代碼僅處理“ <”重定向。 請注意,這只是包裝在while循環中的一小段代碼(shell設計)。 在通過此for循環之后,我得到了:

for(i=0;i<n;i++){
        //printf("extracted token is %s\n",tokens[i]);
        char *ex = "exit";
        char *his = "history";

        if(strcmp(tokens[0],his) == 0 && max_cmd == 0 ){ //get history
            for(j=0;j<counter;j++){
                printf("%i. %s\n",j+1,historic_cmd[j]);         
            }

        }else if(strcmp(tokens[0], his) ==0 && max_cmd == 1){
            for(j=0; j<CMD_MAX;j++){
                printf("%i. %s\n",j+1,historic_cmd[j]);
            }
        }else if(strcmp(tokens[0],ex) == 0){ //exit program
            exit(2);
        }else{                      //forking
            pid = fork();
            if(pid){
                pid=wait(NULL);

                if(infile > 0)
                    dup2(fileno(fd),0);

                if(outfile > 0)
                    dup2(fileno(fd),1);

            }else{
                if(execvp(tokens[0],tokens)){
                    puts(strerror(errno));
                    exit(127);
                }

            }

        }




    } // end of for loop



}//end of while loop for user input

我對為什么它不執行重定向感到困惑。 如果輸入以下內容:

ps > s

它在工作目錄中創建文件,但為空。 我是否錯誤地使用了“ dup2”?

來自“ ps> s”的樣本輸出:

user$>ps > s
ps: illegal option -- >
usage: ps [-AaCcEefhjlMmrSTvwXx] [-O fmt | -o fmt] [-G gid[,gid...]]
      [-u]
      [-p pid[,pid...]] [-t tty[,tty...]] [-U user[,user...]]
   ps [-L]
ps: illegal option -- >
usage: ps [-AaCcEefhjlMmrSTvwXx] [-O fmt | -o fmt] [-G gid[,gid...]]
      [-u]
      [-p pid[,pid...]] [-t tty[,tty...]] [-U user[,user...]]
   ps [-L]
ps: illegal option -- >
usage: ps [-AaCcEefhjlMmrSTvwXx] [-O fmt | -o fmt] [-G gid[,gid...]]
      [-u]
      [-p pid[,pid...]] [-t tty[,tty...]] [-U user[,user...]]
   ps [-L]
found > !!!!!----------------------------------------------------

您替換進程的stdinstdout (在if(pid) ),這不會影響您的孩子。 dup2內容移到execvp之前的else分支。

這是在C中進行重定向時應使用的流程。

int file_stdin;
int file_stdout;
int file_stderr;

/*--------------------------------------------------------------------
; code to set infile and outfile I'll assume that if redirection isn't
; being used, the appropriate filehandle will be -1.
;
; file_stdin  should be a file that is opened using O_RDONLY.
; file_stdout should be a file that is opened using O_WRONLY.
; file_stdout should be a file that is opened using O_WRONLY.
;
; Be wary of using a file descriptor opened for both reading and writing
; and using the same file descriptor---it will probably not do what you
; expect it to do.
;
; Also note, you can also use file descriptors from the pipe() system
; call (to allow piping from command to command) or network sockets, or
; anything that can be read/written from a file descriptor.
;---------------------------------------------------------------------*/

pid_t child;

child = fork();
if (child < 0)
  report_error(errno);
else if (child > 0)
{
  int status;
  int rc;

  /*------------------------------------------------------------------
  ; wait for the child process.  More code needs to be added to
  ; descriptor errors and what not, but that's left as an exercise for
  ; the reader
  ;-----------------------------------------------------------------*/

  rc = wait(&status);
}
else
{
  /*--------------------------------------------------------------------
  ; child process.  Set up redirection if required.  Once the rediretion
  ; has been set (using dup2()), the original file descriptor is closed   
  ; as it is no longer needed and just wastes file descriptors in the   
  ; child process.
  ;-------------------------------------------------------------------*/

  if (file_stdin > -1)
  {
    dup2(file_stdin,STDIN_FILENO);
    close(file_stdin);
  }

  if (file_stdout > -1)
  {
    dup2(file_stdout,STDOUT_FILENO);
    close(file_stdout);
  }

  if (file_stderr > -1)
  {
    dup2(file_stderr,STDERR_FILENO);
    close(file_stderr);
  }

  /*------------------------------------------------------------------
  ; close any unneeded open files here ... 
  ;------------------------------------------------------------------*/

  /*------------------------------------------------------------------
  ; PROGRAM, ARGV and ENVP need to be replaced with appropriate values
  ;------------------------------------------------------------------*/

  if(execve(PROGRAM,ARGV,ENVP) < 0)
    _exit(EXIT_FAILURE);
}

顯然,上面的代碼需要適應您的項目,但這只是一般流程。

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM