Перенаправление вывода в файл в C

Я написал базовую оболочку в C для выполнения основных команд, она будет выполнять команды ls, ls -al, ls -al | more и т.п.

я хочу выполнить следующую команду в моей оболочке. лайк;

ls -al > a.txt

это даст мне a.txt файл, который содержит вывод ls -al процесс. я нашел одно решение, это меняет команду в моей оболочке, как [command1] | tee [filename], в этом случае это изменится ls -al > a.txt в ls -al | tee a.txt, но этот процесс дает вывод в файл и терминал, а также. как остановить вывод на печать в терминале.

или есть какое-то лучшее решение, чем использование команды tee. заранее спасибо...

3 ответа

Решение

Это результат моего тестирования с dup2

Более тонкий момент - запоминание fflush в нужное время:) В противном случае вы получите очень удивительные результаты.

Также предпочитаю fileno вместо жесткого кодирования 1 (Стандартный вывод) 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);

Конечно, вы должны иметь некоторую обработку ошибок.

Другие вопросы по тегам