标准输入流,输出流,错误流 以及 重定向 的原理
标准输入流、输出流、错误流在操作系统与C语言中的表达1. 操作系统层面Linux/Unix在操作系统层面标准输入、标准输出和标准错误流通过文件描述符File Descriptor来标识流类型文件描述符 (fd)默认设备标准输入0通常是终端/键盘标准输出1通常是终端/屏幕标准错误2通常是终端/屏幕这些文件描述符本质上是进程打开的文件句柄可以指向终端设备、普通文件或管道。2. C语言层面在C语言中标准库stdio.h提供了三个预定义的FILE*指针#includestdio.h// 三种标准流指针stdin// 标准输入流stdout// 标准输出流stderr// 标准错误流对应关系C语言流文件描述符说明stdin0fgets()、scanf() 从此读取stdout1printf()、puts() 往此写入有缓冲stderr2fprintf(stderr, …) 往此写入无缓冲示例代码#includestdio.h#includeunistd.h// for close()intmain(){// C语言方式使用标准流fprintf(stdout,正常输出\n);fprintf(stderr,错误信息\n);// 获取底层文件描述符intfd_infileno(stdin);// 0intfd_outfileno(stdout);// 1intfd_errfileno(stderr);// 2// 直接使用系统调用写入write(fd_out,hello\n,6);return0;}3. stdout 与 stderr 的关键区别特性stdoutstderr缓冲有缓冲行缓冲或全缓冲无缓冲立即写出用途程序正常输出错误/诊断信息重定向可单独重定向可单独重定向重定向示例# 只重定向 stdout 到文件./progoutput.txt# 只重定向 stderr 到文件./prog2error.log# 将 stdout 和 stderr 都重定向到同一文件./progall.txt214. 重定向的实现原理./prog output.txt这个重定向是如何实现的这是通过shell 在调用程序前修改文件描述符实现的。核心机制如下是 shell 的语法不是程序本身的行为shell 会打开目标文件把文件的文件描述符设为 1标准输出再执行你的程序程序看到的 fd 1 已经不是终端而是文件了用系统调用还原./prog output.txt// shell 实际做的事简化版pid_tpidfork();if(pid0){// 子进程执行重定向intfdopen(output.txt,O_WRONLY|O_CREAT|O_TRUNC,0644);dup2(fd,1);// 让 fd 1 指向文件close(fd);// 执行目标程序execvp(./prog,argv);}dup2(fd, 1)的作用是让文件描述符 1 和 fd 指向同一个文件。之后程序里所有的printf()、write(1, ...)都写到文件里了。验证实验// test.c#includestdio.h#includeunistd.hintmain(){printf(hello\n);// 走 stdout (fd 1)write(1,world\n,6);// 直接写 fd 1return0;}$ gcc test.c-otest$ ./test# 输出到屏幕$ ./testout.txt# 输出到文件$catout.txt# 输出hello\nworld\n程序代码完全没变变的只是 shell 在 exec 前把 fd 1 换成了文件。5. 重定向的恢复问题dup2(fd, 1)后何时重新把 1 指回标准输出流不会自动指回来——重定向的影响会持续到这个进程结束。dup2(fd, 1)之后这个进程的 fd 1 就永远指向文件了直到进程退出所有文件描述符自动关闭进程自己再用dup2()改回去shell 为什么能恢复因为 shell 是在子进程中做重定向------------------ | 父 shell 进程 | | fd 1 终端 | ← 从来不会被改变 ------------------ | fork() | v ------------------ | 子进程 | | fd 1 终端 | ← 刚 fork 时继承自父进程 | dup2(fd, 1) | | fd 1 文件 | ← 只有子进程被改了 | exec(./prog) | ------------------ | 退出 v 子进程销毁fd 全部关闭父 shell 的 fd 1 一直是终端不受影响。在程序内部恢复的方法如果想在程序内部恢复可以在重定向前先保存原来的 fd 1#includestdio.h#includeunistd.h#includefcntl.hintmain(){// 保存原始标准输出intsaved_stdoutdup(1);// 复制 fd 1得到比如 fd 3// 重定向到文件intfdopen(output.txt,O_WRONLY|O_CREAT|O_TRUNC,0644);dup2(fd,1);close(fd);// ... 现在 printf 都写到文件 ...printf(写入文件\n);// 恢复标准输出dup2(saved_stdout,1);// 把 fd 1 改回原来的终端close(saved_stdout);// ... 现在 printf 又写到屏幕了 ...printf(回到屏幕\n);return0;}6. 总结场景何时恢复shell 重定向 (./prog out.txt)子进程退出父进程从未受影响程序内自己dup2重定向不会自动恢复需要手动dup2(saved_fd, 1)通过理解文件描述符、标准流和重定向机制可以更好地掌握Linux/Unix系统编程和C语言中I/O操作的本质。
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2626900.html
如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!