繁体   English   中英

在C中将系统的使用更改为Fork和exec

[英]Changing the use of system to Fork and exec in C

我有一个正在使用的C程序,它需要一定数量的输入并将其作为系统命令运行。 其余的将传递给shell执行。 但是有人建议,我应该尝试使用fork和exec来运行命令。 我为如何做到这一点而感到困惑。

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

#define MAX_BUFFER 1024                        // max line buffer
#define MAX_ARGS 64                            // max # args
#define SEPARATORS " \t\n"                     // token sparators

extern char **environ;
/*******************************************************************/

int main (int argc, char ** argv)
{
char linebuf[MAX_BUFFER];                  // line buffer
char cmndbuf[MAX_BUFFER];                  // command buffer
char * args[MAX_ARGS];                     // pointers to arg strings
char ** arg;                               // working pointer thru args
char * prompt = "==>" ;                    // shell prompt

// keep reading input until "quit" command or eof of redirected input 

while (!feof(stdin)) { 

// get command line from input


    fputs (prompt, stdout);                // write prompt
    fflush(stdout);
    if (fgets(linebuf, MAX_BUFFER, stdin )) { // read a line

// tokenize the input into args array

        arg = args;
        *arg++ = strtok(linebuf,SEPARATORS);   // tokenize input
        while ((*arg++ = strtok(NULL,SEPARATORS)));
                                           // last entry will be NULL 

        if (args[0]) {                     // if there's anything there

            cmndbuf[0] = 0;                // set zero-length command string

// check for internal/external command 

            if (!strcmp(args[0],"clr")) {  // "clr" command
                strcpy(cmndbuf, "clear");
            } else
            if (!strcmp(args[0],"cd"))
            {
                int ret;
                if (!args[1])
                    strcpy(cmndbuf, "pwd");
                ret = chdir(args[1]);
                strcpy(cmndbuf, "pwd");


            }else
            if (!strcmp(args[0],"dir")) {  // "dir" command
                strcpy(cmndbuf, "ls -al ");
                if (!args[1])
                    args[1] = ".";         // if no arg set current directory
                strcat(cmndbuf, args[1]);
            } else
            if (!strcmp(args[0],"environ")) { // "environ" command
                char ** envstr = environ;
                while (*envstr) {            // print out environment
                    printf("%s\n",*envstr);
                    envstr++;
                }                            // (no entry in cmndbuf)
            } else
            if (!strcmp(args[0],"quit")) {   // "quit" command
                break;
            } else {                         // pass command on to OS shell
                int i = 1;
                strcpy(cmndbuf, args[0]);
                while (args[i]) {
                    strcat(cmndbuf, " ");
                    strcat(cmndbuf, args[i++]);
                }
            }

// pass any command onto OS

            if (cmndbuf[0])
                system(cmndbuf);
        }
    }
}
return 0; 
}

我假设您有一个POSIX系统,例如GNU / Linux。

这是一个非常普遍的问题。 第一个答案是研究诸如GNU Libc之类的免费C库中system功能的实现。 Unix上的许多好书也涵盖了这个问题。

作为提示, system使用fork -ing,然后在shell /bin/sh的子进程execve中工作

Brian Kernighan和Rob Pike 在UNIX编程环境中“系统”函数实现的示例。

#include <signal.h>
system(s) /* выполнить командную строку s */
char *s;
{
  int status, pid, w, tty;
  int (*istat)(), (*qstat)();
  extern char *progname;
  fflush(stdout);
  tty = open("/dev/tty", 2);
  if (tty == 1) {
    fprintf(stderr, "%s: can't open /dev/tty\n", progname);
    return 1;
  }
  if ((pid = fork()) == 0) {
    close(0); dup(tty);
    close(1); dup(tty);
    close(2); dup(tty);
    close(tty);
    execlp("sh", "sh", " c", s, (char *) 0);
    exit(127);
  }
  close(tty);
  istat = signal(SIGINT, SIG_IGN);
  qstat = signal(SIGQUIT, SIG_IGN);

  while ((w = wait(&status)) != pid && w != 1);
  if (w == 1)
    status = 1;
  signal(SIGINT, istat);
  signal(SIGQUIT, qstat);
  return status;
}

请注意,这本书是在C标准最终确定之前编写的,因此它不使用原型。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM