#c #file #dup2
#c #файл #dup2
Вопрос:
Я пытаюсь изучить dup2
и переключить стандартный вывод на файл, а не на терминал. Это пример, который работает везде, но не уверен, почему он не работает у меня. Я не думаю, что мне нужен fork(), потому что мне не нужен другой процесс для выполнения только инструкции print в file.
Где вызывается функция:
int main(int argc, char **argv){
char *something = "hello";
saveHistoryToFile(something);
}
// Это функция. Существует история имен файлов .txt
void saveHistoryToFile(char *history){
int fw = open("history.txt",O_WRONLY | O_APPEND);
dup2(fw, 1);
printf("%s", history);
}
ОШИБКА: он печатает в терминал, а не в файл!
Комментарии:
1. Что именно означает «не работает для меня»? Не компилируется? Сбой во время выполнения? Выдает неверный результат?
2. Для меня все в порядке, пока
"history.txt"
существует, в противном случае добавьтеO_CREAT
. Проверка ошибокperror
или запуск подstrace
должны показать, в чем проблема.3. Делает
fflush(stdout);
ли это послеprintf()
исправления?4. он не печатает в файл, а просто печатает в терминале @JohnBollinger
5. Вы не проверяете возвращаемые значения ваших вызовов функций, чтобы определить, успешны ли они. Велика вероятность, что ваш вызов
open()
завершается неудачно, и, как следствие, ваш вызов todup2()
также завершается неудачно. Это оставило бы вас со стандартным выходом, все еще подключенным к терминалу (или к чему-либо еще, к чему он был подключен изначально).
Ответ №1:
Ваш код с проверкой ошибок:
#include <stdio.h>
#include <unistd.h>
#include <fcntl.h>
int saveHistoryToFile(char *history);
int main(int argc, char **argv){
char *something = "hello";
if(0>saveHistoryToFile(something)) return 1;
if(0>fclose(stdout)) return perror("fclose"),-1;
}
int saveHistoryToFile(char *history){
int fw = open("history.txt",O_WRONLY | O_APPEND /*|O_CREAT, 0640*/ );
if (0>fw) return perror("open"),-1;
if (0>dup2(fw, 1)) return perror("dup2"),-1;
if (0>(printf("%s", history))) return perror("printf"),-1;
}
При первом запуске я получаю «открыть: нет такого файла или каталога», потому что у меня нет "history.txt"
в моем текущем каталоге.
Если я добавлю его или раскомментирую O_CREAT, 0640
, он отлично работает на моей машине.
Конечно, вы можете столкнуться с другими проблемами (например, EPERM
), но perror
s должен дать вам подсказку.
Комментарии:
1. Это дает мне отказ в разрешении
2. при открытии в функции
3. @BassamMetwally И вы не можете устранить это? Какой у вас текущий каталог, каковы его разрешения (попробуйте запустить
system("stat "$PWD" history.txt");
).4. @BassamMetwally Вам также нужен аргумент mode с
O_CREAT
флагом. Я забыл об этом.5. ДА! это сработало, СПАСИБО. Я только что запустил chmod, чтобы разрешить запись и чтение.