перенаправление вывода в файл в C
я закодировал базовую оболочку в C для выполнения основных команд, он будет выполнять команды ls
, ls -al
, ls -al | more
etc.
Я хочу выполнить следующую команду в своей скорлупе. как ;
ls -al > a.txt
это даст мне a.txt
файл, который содержит выходные данные
3 ответов
это результат моего тестирования с dup2
более тонкий момент-это запоминание fflush в нужное время :) в противном случае вы получите очень удивительные результаты.
кроме того, предпочитают fileno
вместо hardcoding 1
(stdout) 2
(stderr).
перенаправления stdin
оставлено в качестве упражнения для читателя
#include <stdio.h>
#include <stdlib.h>
#include <fcntl.h>
#include <unistd.h>
int main(int argc, const char *argv[])
{
int out = open("cout.log", O_RDWR|O_CREAT|O_APPEND, 0600);
if (-1 == out) { perror("opening cout.log"); return 255; }
int err = open("cerr.log", O_RDWR|O_CREAT|O_APPEND, 0600);
if (-1 == err) { perror("opening cerr.log"); return 255; }
int save_out = dup(fileno(stdout));
int save_err = dup(fileno(stderr));
if (-1 == dup2(out, fileno(stdout))) { perror("cannot redirect stdout"); return 255; }
if (-1 == dup2(err, fileno(stderr))) { perror("cannot redirect stderr"); return 255; }
puts("doing an ls or something now");
fflush(stdout); close(out);
fflush(stderr); close(err);
dup2(save_out, fileno(stdout));
dup2(save_err, fileno(stderr));
close(save_out);
close(save_err);
puts("back to normal output");
return 0;
}
не используйте pipe, когда вывод должен перейти в файл.
когда вы вилка ребенка, чтобы запустить ls
команда, обратите внимание на перенаправление и откройте Файл; затем используйте dup2()
(или close()
и dup()
), Так что файловый дескриптор теперь является стандартным выходом для ребенка; вы закрываете дублированный файловый дескриптор-тот, который возвращается open()
; затем вы выполнить ls
как обычно, ее стандартный вывод теперь отправляется в файл.
заметьте что вы делаете и/о не-трубы перенаправление после раздвоения, не раньше. Трубы должны быть настроены перед разветвлением, но другое перенаправление ввода-вывода-нет.
прежде чем позвонить execve(2)
во вновь созданный процесс для выполнения команды, вы можете перенаправить стандартный ввод или вывод через dup2(2)
системный вызов:
/* redirect stdout to a file */
dup2(1, some_open_file_descriptor);
конечно, вам нужно иметь некоторую обработку ошибок.