gpt4 book ai didi

c - 在C中重定向子进程的文件输出

转载 作者:太空宇宙 更新时间:2023-11-04 00:02:57 40 4
gpt4 key购买 nike

我要么完全忘记了我想知道的关于 C 的一切,要么出了什么大问题。我想将子进程(stdout 和 stderr)的输出重定向到一个文件。我这样做:

if ((pid = fork()) == 0)
{
get_host_date(today) ;
int fd = open(log_filename, O_RDWR | O_CREAT | O_APPEND,
S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP | S_IROTH) ;
dup2(fd, STDOUT_FILENO) ; // make stdout go to file
dup2(fd, STDERR_FILENO) ; // make stderr go to file
fprintf(stderr, "This is a message") ;

if ((status = execv(command, argv)) == -1)
{
printf("\n\nError: execve failed for %s", command) ;
perror("\nexecve failure: ") ;
exit(EXIT_FAILURE) ;
}
close(STDOUT_FILENO) ;
close(STDERR_FILENO) ;
}

日志文件按指定创建,但输出未转到该文件。我做了一些测试输出 (fprintf(stderr, "This is a message") ;),但这并没有出现在任何地方。Wenn 我检查了变量 fd,我看到它的值为 1(紧接在 open 之后)。但是 1 不应该是预定义的输出文件描述符吗?

有人能帮帮我吗?我尝试了一切,但没有结果。

先谢谢你了最好的祝愿约尔格

P.S.:我正在使用 RHEL 和 GNU-C。

好吧,这里有一个压缩代码,但我无法使用:

int main(int   argc,
char* argv[])
{
PROC_REC proc ;

strcpy(proc.command, "/home/islk/entw/v0816/exe/islk_server") ;
strcpy(proc.args[0], "ISLK_DB1_SERV01") ;
strcpy(proc.args[1], "") ;
strcpy(proc.env[0], "ISLKSERVER_NR=5") ;
strcpy(proc.env[1], "") ;

ISLK_start_single_process(&proc) ;

exit(EXIT_SUCCESS) ;
}

static long ISLK_start_single_process(PROC_REC *prec_ptr)
{
long i=0 ;
int status ;
char *argv[16] ;
char command[256], log_filename[128], today[DB_DATE_DOM] ;
pid_t pid ;

/* Set arguments */
for (i=0; i<16; i++)
{
if (strcmp(prec_ptr->args[i], "") != 0)
argv[i] = prec_ptr->args[i] ;
else
argv[i] = NULL ;
}

/*******************/
/* Set environment */
/*******************/
for (i=0; i<16; i++)
{
if (strcmp(prec_ptr->env[i], "") != 0)
putenv(prec_ptr->env[i]) ;
}

/*****************/
/* Start process */
/*****************/
if ((pid = fork()) == 0)
{
get_host_date(today) ;
bs_create_filename(log_filename, "islk$log", "", "%s_%8.8s.log",
prec_ptr->args[0], today) ;
int fd = open(log_filename, O_RDWR | O_CREAT | O_APPEND,
S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP | S_IROTH) ;
int fdo = dup2(fd, STDOUT_FILENO) ; // make stdout go to file
int fde = dup2(fd, STDERR_FILENO) ; // make stderr go to file
close(fd) ;
fprintf(stdout, "This is a message") ;
if ((status = execv(command, argv)) == -1)
{
printf("\n\nError: execv failed for %s", command) ;
perror("\nexecv failure: ") ;
exit(EXIT_FAILURE) ;
}
close(STDOUT_FILENO) ;
close(STDERR_FILENO) ;
}
else if (pid < 0)
{
printf("\n\nError: fork failed for %s", prec_ptr->args[0]) ;
perror("\nfork failure: ") ;
return ERROR ;
}
else
{
printf("\nProcess %d started for %s", pid, prec_ptr->args[0]) ;
prec_ptr->pid = pid ;
}

return NO_ERROR ;
}

尝试了以下方法:

dup2(fd, STDERR_FILENO) ;
dup2(fd, STDOUT_FILENO) ;

-> 到 stderr 的消息写在文件中,而不是到 stdout

dup2(fd, STDOUT_FILENO) ;

-> 到 stderr 的消息写在终端中,到 stdout 无处可去

dup2(fd, STDOUT_FILENO) ;
dup2(STDERR_FILENO, STDOUT_FILENO) ;

-> 到 stderr 的消息被写在文件中,到终端上的 stdout

dup2(fd, STDERR_FILENO) ;
dup2(STDOUT_FILENO, STDERR_FILENO) ;

-> 到 stderr 的消息被写在文件中,到终端上的 stdout

-> stdout 似乎有问题!

最佳答案

我认为您几乎已经通过检测到 fd == 1 “紧接在 open() 之后”自行回答了这个问题。根据the dup2() man page :

The dup2() system call performs the same task as dup(), but instead of using the lowest-numbered unused file descriptor, it uses the descriptor number specified in newfd. If the descriptor newfd was previously open, it is silently closed before being reused.

因此,如果 fd == 1 并且您调用了 dup2( fd, STDOUT_FILENO ); 那么 STDOUT_FILENO(也是 1)被关闭,您的描述符也是如此重定向文件。如果您无法找出为什么 stdout 在您的程序环境中被关闭,也许您可​​以在调用dup2() 之前做一个测试:

if( fd != STDOUT_FILENO )
dup2( fp, STDOUT_FILENO );

至少这应该可以作为检查我是否正确的技巧。

关于c - 在C中重定向子进程的文件输出,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/35776427/

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