gpt4 book ai didi

c - 迷失在多个 Fork()、Pipe() 和 Select()

转载 作者:行者123 更新时间:2023-12-04 07:01:00 25 4
gpt4 key购买 nike

希望我能在这里找到一些帮助,因为我开始放弃了。请注意,因为这是一项家庭作业,因此为什么它可能很愚蠢。

语境:
必须编写一些将被 shell 执行的东西:
登录 [--tick n] cmd [args] [, cmd [args]]...

基本上,它是一个同时运行多个程序的程序。

约束:
每个输出行都必须以它的命令编号开头,格式为 printf "%d: %s"
IE:
0:第一个命令的第一行。
0:第一个命令的第二行。
1:第二条命令的第一行。
0:第一条命令的第三行。
1:第二条命令的第二行。

如果已指定刻度,则在 n 秒内未发送任何输出时,系统将打印一个句点。
必须使用 Select()
如果最后一个输出是一个句点,系统不会打印另一个句点。

现在这是我的问题!我能够使用单个命令使其工作。当我尝试使其成为多个命令时,我似乎失败了。我相信这可能是我的方法。也许你们中的一些人能够帮助我。

这是我尝试使用 Multiple Cmd 的方法。我可能完全错了,因此我需要帮助:

#include <string.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <errno.h>
#include <sys/types.h>
#include <sys/time.h>
#include "readline.h"

// Reference: http://www.gnu.org/s/libc/manual/html_node/Waiting-for-I_002fO.html
int input_timeout (int filedes, unsigned int seconds)
{
fd_set set;
struct timeval timeout;

/* Initialize the file descriptor set. */
FD_ZERO (&set);
FD_SET (filedes, &set);

/* Initialize the timeout data structure. */
timeout.tv_sec = seconds;
timeout.tv_usec = 0;

/* select returns 0 if timeout, 1 if input available, -1 if error. */
return (select (FD_SETSIZE,&set, NULL, NULL, &timeout));
}

/* Fetches the number of commands in parameters. Number of Commas + 1 */
int getNbCmd(char ** argv)
{
int nbCmd = 1;
int i;
while(argv[i] != '\0')
{
if(strcmp(argv[i], ",") == 0)
nbCmd++;
i++;
}

return nbCmd;
}


/* Fills the Command Array */
void getCommandes(char *** tbCmd, int argc, char ** argv)
{
int indexArgv = 1;
int indexCmd = 0;
int indexTbCmd = 0;

char ** cmd = (char **)malloc(argc*sizeof(char *));

if(strcmp(argv[indexArgv], "--tick") == 0)
indexArgv = 3;

while (indexArgv < argc)
{

if(strcmp(argv[indexArgv], ",") == 0)
{
cmd[indexCmd] = (char *) 0;
tbCmd[indexTbCmd] = cmd;
free(cmd);
cmd = (char **)malloc(argc*sizeof(char *));

indexTbCmd++;
indexCmd = 0;
}
else
{
char * arg;
arg = argv[indexArgv];
cmd[indexCmd] = arg;

indexCmd++;
}
indexArgv++;
}

cmd[indexCmd] = (char *) 0;
tbCmd[indexTbCmd] = cmd;
free(cmd);
}


int main (int argc, char ** argv)
{
int nbCmds = getNbCmd(argv);
int tick = -1;

char *** tbCmd = (char ***) malloc (nbCmds*sizeof(char **));

if(strcmp(argv[1], "--tick") == 0)
tick = atoi(argv[2]);

getCommandes(tbCmd, argc, argv);

int i;

pid_t pidM[nbCmds];
int p[nbCmds][2];

for (i = 0;i < nbCmds;i++)
{
if ( pipe( p[i] ) != 0 ){ perror( "pipe()" ); exit(1); }

// fork() to get child process
pidM[i] = fork();

if ( pidM[i] < 0 ){ perror( "fork()" ); exit(1); }
else if (pidM[i] == 0)
{
close(p[i][0]);
dup2(p[i][1], STDOUT_FILENO);

int ret;
ret = execvp(tbCmd[i][0], tbCmd[i]);
}
else
{
close(p[i][1]);

char * buffer;
int retval = 1;
int pntAfficher = 0; //Boolean for Period Printing
/* select returns 0 if timeout, 1 if input available, -1 if error. */
if(tick >= 0)
retval = input_timeout(p[i][0], tick);
if (retval == 0 && pntAfficher == 0)
{
printf(".\n");
pntAfficher = 1;
}

buffer = readline(p[i][0]);
while(buffer[0] != '\0')
{
printf("%d: %s",i, buffer);
free(buffer);

/* select returns 0 if timeout, 1 if input available, -1 if error. */
if(tick >= 0)
retval = input_timeout(p[i][0], tick);
if (retval == 0 && pntAfficher == 0)
{
printf(".\n");
pntAfficher = 1;
}
buffer = readline(p[i][0]);
pntAfficher = 0;

free(buffer);
}
}
}
free(tbCmd);

}

最佳答案

您需要在一个循环中 fork 所有子级,然后在第二个循环中,从所有子级中读取所有数据。

您想要并行收听所有 child 的输出(这就是 select 为您所做的),但所有 child 都需要运行才能使其工作。您当前的 for循环产生一个 child ,然后转到 select ......这不是它的正确位置。

关于c - 迷失在多个 Fork()、Pipe() 和 Select(),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/1851020/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com