首先要知道 FILE *stin 和 STDIN_FILENO 的区别。
stdin类型为 FILE*
STDIN_FILENO类型为 int
使用stdin的函数主要有:fread、fwrite、fclose等,基本上都以f开头
使用STDIN_FILENO的函数有:read、write、close等
操作系统一级提供的文件API都是以文件描述符来表示文件。STDIN_FILENO就是标准输入设备(一般是键盘)的文件描述符。
标准C++一级提供的文件操作函数库都是用FILE*来表示文件,stdin就是指向标准输入设备文件的FILE*。
如下:
参考stdin(3)的man手册:Name
stdin, stdout, stderr - standard I/O streams
Synopsis#include
extern FILE *stdin;
extern FILE *stdout;
extern FILE *stderr;
stdin / stdout / stderr 分别是指向stream的FILE型的指针变量。当程序启动时,与其结合的整型文件描述符(fd)分别是0,1,2。
STDIN_FILENO / STDOUT_FILENO / STDERR_FILENO 是在< unistd.h>文件中定义的预编译macro。其值是0,1,2。
(通过freopen(3),可以改变与这3个文件描述符(fd)结合的stream值)
也就是说,程序启动时: FILE * stdin / stdout / stderr 对应的 文件描述符(fd)分别是 STDIN_FILENO(0) / STDOUT_FILENO(1) / STDERR_FILENO(2) 。但是,可以通过FILE *freopen(const char *path, const char *mode, FILE *stream); 来改变,使文件描述符(fd)对应到其他的stream上。
STDIN_FILENO / STDOUT_FILENO / STDERR_FILENO这三个文件描述符(fd),可通过freopen(),来改变与之结合的stream。 (结合到那个文件stream上,那个文件stream就是变成 标准输入输出)
标准输入stdin标准输出stdouterr输出stderr 分别改变。
例程序
#include
#include
#include
#include
#define CHANGE_BY_FREOPEN /* 需要注释掉,进行切换 */
int main(int argc,char**argv){
char buf[]="hello,world\n";
#ifdef CHANGE_BY_FREOPEN
freopen("stdout_text.txt","w",stdout);
//freopen("stderr_text.txt","w",stderr);/* stderr */
#endif
printf("%s",buf);
fwrite(buf,strlen(buf), 1,stdout);
write(STDOUT_FILENO,&buf,strlen(buf));
perror("error out");/* stderr */
return(0);
}/* stdouttest.c */
编译命令,生成可执行文件 stdouttest$ gcc -o stdouttest stdouttest.c
①当 #define CHANGE_BY_FREOPEN 被注释掉,无效的时候: 终端shell窗口中的输出结果是:
$ ./stdouttest
hello,worldhello,worldhello,worlderror out: No error
$
注:3个(hello,world + 换行符)。1个error的输出标准输入输出,都还是shell窗口。
②当 #define CHANGE_BY_FREOPEN ,有效的时候: 终端shell窗口中的输出结果是:
$ ./stdouttest
error out: No error
$
注:shell窗口中,仅有stderr的输出。
但是,会创建一个名为stdout_text.txt的文件,该文件中的内容是:hello,worldhello,worldhello,world
注:3个(hello,world + 换行符)。 仅标准输出,从shell窗口,改变为stdout_text.txt的文件了。
//freopen("stderr_text.txt","w",stderr);/* stderr */如果有效的话:①shell中,什么也不输出。②stdout_text.txt 中,输出3个(hello,world + 换行符)。③stderr_text.txt 中,输出1个(error out: No error)。