FILE *stdout 和 STDOUT_FILENO 的区别

转自:http://blog.sina.com.cn/s/blog_9f1496990100y3st.html

首先要知道 FILE *stdout 和 STDOUT_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 <stdio.h>
extern FILE *stdin;
extern FILE *stdout;
extern FILE *stderr;
...
On program startup, the integer file descriptors associated with the streams stdin, stdout, and stderr are 0, 1, and 2, respectively. The preprocessor symbols STDIN_FILENO, STDOUT_FILENO, and STDERR_FILENO are defined with these values in <unistd.h>. (Applying freopen(3) to one of these streams can change the file descriptor number associated with the stream.)
...

stdin / stdout / stderr
分别是指向stream的FILE型的指针变量。
当程序启动时,与其结合的整型文件描述符(fd)分别是0,1,2。

STDIN_FILENO / STDOUT_FILENO / STDERR_FILENO
是在

#include<stdio.h>
#include<unistd.h>
#include<string.h>
#include<errno.h>

#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);
#endif

printf("%s",buf);
fwrite(buf,strlen(buf), 1,stdout);
write(STDOUT_FILENO,&buf,strlen(buf));
perror("error out");

return(0);
}

编译命令,生成可执行文件 stdouttest

$ gcc -o stdouttest stdouttest.c


当 #define CHANGE_BY_FREOPEN 被注释掉,无效的时候:
终端shell窗口中的输出结果是:

$ ./stdouttest
hello,world
hello,world
hello,world
error 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,world
hello,world
hello,world

注:
3个(hello,world + 换行符)。
仅标准输出,从shell窗口,改变为stdout_text.txt的文件了。

//freopen(“stderr_text.txt”,”w”,stderr);
如果有效的话:
①shell中,什么也不输出。
②stdout_text.txt 中,输出3个(hello,world + 换行符)。
③stderr_text.txt 中,输出1个(error out: No error)。

在Linux中,标准输入(stdin)、标准输出(stdout)和标准错误(stderr)是预定义的文件描述符,它们分别是0、1和2。如果我们需要重定向这些文件描述符,可以使用以下命令: ``` command < input_file.txt # 将文件input_file.txt中的内容作为标准输入 command > output_file.txt # 将标准输出写入文件output_file.txt中 command 2> error_file.txt # 将标准错误输出写入文件error_file.txt中 ``` 如果需要恢复这些文件描述符到原始状态,我们可以使用dup2()函数。它可以将一个文件描述符复制到另一个文件描述符的位置上,从而实现重定向的撤销。以下是一个示例代码: ```c #include <stdio.h> #include <unistd.h> #include <fcntl.h> int main() { int fd_in = open("input_file.txt", O_RDONLY); int fd_out = open("output_file.txt", O_WRONLY | O_CREAT | O_TRUNC, 0666); int fd_err = open("error_file.txt", O_WRONLY | O_CREAT | O_TRUNC, 0666); // 重定向标准输入、输出和错误输出 dup2(fd_in, STDIN_FILENO); dup2(fd_out, STDOUT_FILENO); dup2(fd_err, STDERR_FILENO); // 执行一些需要重定向的代码 printf("Hello, world!\n"); fprintf(stderr, "This is an error message.\n"); // 恢复标准输入、输出和错误输出 dup2(STDIN_FILENO, fd_in); dup2(STDOUT_FILENO, fd_out); dup2(STDERR_FILENO, fd_err); // 关闭文件描述符 close(fd_in); close(fd_out); close(fd_err); return 0; } ``` 在上面的代码中,我们首先打开三个文件描述符,并将它们分别复制到标准输入、输出和错误输出的位置上。然后我们执行一些需要重定向的代码,最后使用dup2()函数将文件描述符恢复到原始状态。最后关闭文件描述符,释放资源。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值