dup2 не переключается на файл?

#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() завершается неудачно, и, как следствие, ваш вызов to dup2() также завершается неудачно. Это оставило бы вас со стандартным выходом, все еще подключенным к терминалу (или к чему-либо еще, к чему он был подключен изначально).

Ответ №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, чтобы разрешить запись и чтение.