Los resultados de printf() y system() están en el orden incorrecto cuando la salida se redirige a un archivo




linux child-process (2)

Tengo un programa en C que se compila en un ejecutable llamado myprogram. Esta es su función principal:

int main(int argc, char ** argv) {
  printf("this is a test message.\n");
  system("ls");

  return 0;
}

Cuando ejecuto myprogram > output.txt en un shell de Linux y luego examino output.txt, veo el resultado de ls arriba "este es un mensaje de prueba".

Siento que debería ser al revés. ¿Por qué sucede esto y qué puedo hacer para que aparezca "este es un mensaje de prueba" en la parte superior de output.txt?

Si importa, soy nuevo en C y trabajo en una línea de comandos.


De forma predeterminada, la salida a la salida estándar está en búfer de línea cuando se conecta a un terminal. Es decir, el búfer se vacía cuando está lleno o cuando agrega una nueva línea.

Sin embargo , si stdout no está conectado a un terminal, como lo que sucede cuando redirige la salida de su programa a un archivo, entonces stdout convierte en un búfer completo . Eso significa que el búfer se vaciará y en realidad se escribirá cuando esté lleno o cuando se vacíe explícitamente (lo que sucede cuando el programa sale).

Esto significa que la salida de un proceso separado iniciado desde su código (como lo que sucede cuando llama al system ) probablemente se escribirá primero, ya que el búfer de ese proceso se vaciará cuando finalice ese proceso, que es antes de su propio proceso.

¿Qué sucede cuando se usa la redirección (o las tuberías)?

  1. Su llamada a printf escribe en el búfer stdout .
  2. La función del system inicia un nuevo proceso, que escribe en su propio búfer.
  3. Cuando el proceso externo (iniciado por su llamada al system ) sale, su búfer se vacía y se escribe. Tu propio búfer en tu propio proceso, no se toca.
  4. Su propio proceso termina, y su búfer de stdout se vacía y se escribe.

Para obtener la salida en el orden "correcto" (o al menos esperado), llame a fflush antes de llamar al system , para eliminar explícitamente la salida setbuf , o llame a setbuf antes de cualquier salida para deshabilitar el almacenamiento en búfer por completo.


Está relacionado con el búfer de salida. Logré reproducir el mismo comportamiento. Forzando el color lo hizo por mí.

#include <stdio.h>
#include <stdlib.h>

int main(int argc, char ** argv) {
  printf("this is a test message.\n");
  fflush(stdout);
  system("ls");

  return 0;
}

Antes de añadir el fflush:

$ ./main > foo
$ cat foo
main
main.c
this is a test message.

y después:

$ ./main > foo
$ cat foo
this is a test message.
foo
main
main.c




io-redirection