Linux 7年前

stdin stdout stderr

作者头像 刘宇帅
4016 0

首先我们知道什么是流(stream),流物理设备的输入输出、存储设备上的文件的输入输出都映射到逻辑上的数据流,这是用来消除设备直接差异而抽象的一个概念。 通常情况下每个程序在启动的时候都会定义3个流:stderr、stdout、stderr,分别用来输入、输出、诊断和错误信息的输出,通常他们会被连接到用户终端(tty(4)),但是也可以修改到其他的文件和设备(重定向)。这取决于双亲进程的选择和设置。
Linux的本质是一切皆是文件,输入输出设备也是使用文件的形式存在和管理的,内核启动的时候默认会打开三个I/O设备文件:stdin、stdout、stderr,分别得到的描述符是0,1,2,并且他们分别对应的文件就是 /dev/stdin /dev/stdout /dev/stderr。

shell中的输入输出的使用

我们直接输出字符串,字符串在终端直接显示

> $ echo "test"
test

shell中的输出对象就是标准输出上面的命令跟下面是等价的

> $ echo "test" >/dev/stdout
test

在shell 管道命令中,后面命令的输入就是前面命令的标准输出,例如

> $ echo "test" |grep -v test

我们执行上面的命令那么命令行不会有任何输出因为 echo 的输出目的地不再是 stdout 而是 grep 的输入,而 grep 使用 -v 参数忽略到了 test 字符串,所以 grep 就不会忘标准输出写东西,所以上面命令输出为空。 我们再看下面两条命令

> $ (echo "test" >/dev/stdout) |grep  -v test
> $ (echo "test" >/dev/stderr) |grep -v test
test

为什么第二条会有输出呢?因为在 shell 中标准输出和标准错误的目的地都是命令行窗口,而第二行展示的 test 是标准错误的输出结果,最主要的是管道命令会把前一条命令的标准输出作为下一条命令的输入,而标准错误输出还是作为错误输出到目的地。
我们这里再说下文件重定向。

> $ echo "test"> /tmp/stdout.log
> $ cat /tmp/stdout.log
test

我们把 echo 输出重定向到 /tmp/stdout.log,可以看到终端中没有任何输出,而 /tmp/stdout.log 中可以看到相关内容。
文件重定向同样可以修改标准错误输出目的地

> $ (echo "test" >/dev/stderr) 2> /tmp/stderr.log
> $ cat /tmp/stderr.log
test

我们可以看到终端中没有任何输出,而/tmp/stderr.log 中可以看到相关输出。 输入重定向

> $ cat < /tmp/stdout.log
test

我们知道 cat 的使用方法是 cat filepath,而上面我们使用输入从定向,把/tmpstdout.log作为输入。

Go语言

在 Go 语言中我们在包 os.file 中看到标准输入输出和错误的相关的定义:

var (
    Stdin  = NewFile(uintptr(syscall.Stdin), "/dev/stdin")
    Stdout = NewFile(uintptr(syscall.Stdout), "/dev/stdout")
    Stderr = NewFile(uintptr(syscall.Stderr), "/dev/stderr")
)

Go 中3个流对应的文件和系统对应的文件是一致的。
Go 语言中基本使用:

package main

import "fmt"

func main() {
    fmt.Print("stdout")
    println("stderr")
}

stdout 是行缓冲的

有很多地方说 stdout 是行缓冲的,stderr 是不缓冲的。根据上面的说明我们可以知道他们两个都是文件描述符是没有差别的,所以不存在操作系统层面不存在缓冲的问题。上面说的 stdout 缓冲的问题应该是c/c++语言中对两种输出实现做的优化处理。另外在 Go 代码中没有相关逻辑。

作者头像

刘宇帅

非著名程序员,全栈开发工程师,长期专注系统开发与架构设计。

提示

功能待开通!


暂无评论~

相关文章

linux删除了libc.so.6

不小心删除了libc.so.6 升级系统软件说依赖更高glibc库,然后升级过程中错误删除了libc.so.6。。导致所有的命令均无法执行。 例如执行ll,有如下提示 ll sh: error while loading shared libraries: libc.so.6: cannot open shared object file: No such file or directory libc.so.6介绍 glibc是GNU发布的libc库,即c运行库。glibc是linux系统中最底层的api,几乎其它任何运行库都会依赖于glibc。 sh bash这些都是依赖与glibc库的,所

Linux命令file

file介绍 file命令用来探测文件的类型 用法 -b:列出辨识结果时,不显示文件名称; -c:详细显示指令执行过程,便于排错或分析程序执行的情形; -f&lt;名称文件&gt;:指定名称文件,其内容有一个或多个文件名称时,让file依序辨识这些文件,格式为每列一个文件名称; -L:直接显示符号连接所指向的文件类别; -m&lt;魔法数字文件&gt;:指定魔法数字文件; -v:显示版本信息; -z:尝试去解读压缩文件的内容。 file源码编译安装 github库 以安装5.25为例 wget https://github.com/file/file/archive/FILE5_25

Linux命令xargs

介绍 xargs 是给命令传递参数的一个过滤器,也是组合多个命令的一个工具。 xargs 可以将管道或标准输入(stdin)数据转换成命令行参数,也能够从文件的输出中读取数据。 xargs 也可以将单行或多行文本输入转换为其他格式,例如多行变单行,单行变多行。 xargs 默认的命令是 echo,这意味着通过管道传递给 xargs 的输入将会包含换行和空白,不过通过 xargs 的处理,换行和空白将被空格取代。 xargs 是一个强有力的命令,它能够捕获一个命令的输出,然后传递给另外一个命令。 之所以能用到这个命令,关键是由于很多命令不支持|管道来传递参数,而日常工作中有有这个必要,所以就有了

Centos配置软件源

Centos7修改软件源 这里以Centos7为例子,6同样的方法url里相应7修改为6即可 配置remi源 remi源依赖epel源,安装epel源 yum install epel-release 安装remi源 rpm -ivh https://rpms.remirepo.net/enterprise/remi-release-7.rpm rpm --import /etc/pki/rpm-gpg/RPM-GPG-KEY-remi 启用remi源 /etc/yum.repos.d/remi.repo,在[remi]下边的enabled=0改为enabled=1 同时也安装了各版本的php

Linux命令ssh

ssh介绍 SSH是一种网络协议,用于计算机之间的加密登录。 如果一个用户从本地计算机,使用SSH协议登录另一台远程计算机,我们就可以认为,这种登录是安全的,即使被中途截获,密码也不会泄露。 命令参数 -1:强制使用ssh协议版本1; -2:强制使用ssh协议版本2; -4:强制使用IPv4地址; -6:强制使用IPv6地址; -A:开启认证代理连接转发功能; -a:关闭认证代理连接转发功能; -b:使用本机指定地址作为对应连接的源ip地址; -C:请求压缩所有数据; -F:指定ssh指令的配置文件; -f:后台执行ssh指令; -g:允许远程主机连接主机的转发端口; -i:指定身份文件;