跳转至

「I/O 流」管道和重定向

I/O 流

shell 使用 3 种标准「I/O 流」,每种流与一种文件描述符相关联:

  • stdout 是标准输出流,显示来自命令的输出。文件描述符为 1。
  • stderr 是标准错误流,显示来自命令的错误输出。文件描述符为 2。
  • stdin 是标准输入流,向命令提供输入。文件描述符为 0。

输出重定向

使用 >>> 将输出流重定向到文件。如果文件不存在则创建文件;如果文件已存在的话,> 覆盖文件,>> 附加文本到文件。例如:

echo "some text" > foo.txt
echo "append" >> foo.txt

ps >> ps.output

上面是将 stdout 重定向到文件,下面演示对 stderr 的处理:

$  ls /usr  > output.txt 
$  
$  # 报错的输出是 stderr,它不同于 stdout
$  ls /not-exist > output.txt 
ls: cannot access '/not-exist': No such file or directory
$  
$  # stdout 和 stderr 重定向到不同文件
$  ls /usr /not-exist > stdout.txt 2> stderr.txt
$  
$  cat stderr.txt
ls: cannot access '/not-exist': No such file or directory
$  
$  # stdout 和 stderr 都输出到一个文件,下面两条命令效果一样
$  ls /usr /not-exist &> output.txt
$  ls /usr /not-exist > output.txt 2>&1
$  cat output.txt
ls: cannot access '/not-exist': No such file or directory
/usr:
bin
lib
local
...

上例中 2>&1,2 和 1 分别是 stderr 和 stdout,即将 stderr 重定向到 stdout。ls /usr /not-exist > output.txt 2>&1 这一行的意思是将 stdout 重定向到 output.txt,且 stderr 重定向到 stdout,即全部重定向到文件。如果写成 2>1 则表示 stderr 重定向到 1 这个文件,所以有了 2>&1 这样特别的语法。

屏蔽输出

屏蔽输出只需要重定向到 /dev/null,例如:

$  ls /not-exist
ls: cannot access '/not-exist': No such file or directory
$  
$  # 屏蔽 stderr
$  ls /not-exist 2> /dev/null
$  
$  # 屏蔽所有输出
$  ls /usr /not-exist &> /dev/null

输入重定向

我们先来看一个输入重定向的例子:

$  sort <<EOF
> beef
> cheese
> apple
> EOF
apple
beef
cheese

sort 命令用于对问本行进行排序,可以从文件读取文本,也可以从 stdin 读取文本。<<EOF 表示前面的命令将从 stdin 读取文本,接下来终端显示提示符 > 表示用户可以输入文本,最后敲 EOF 结束重定向 (EOF 可以替换为其他字符,开始和结束标记必须保持一样)。

另一个普遍的用途就是一次性写入多行文本到文件,例如写入一个 foo.txt 文件:

cat > foo.txt <<EOF
This is line one.
This is line two.
EOF

# 另一种写法
cat <<EOF > foo.txt
This is line one.
This is line two.
EOF

管道

管道符号 |(英文名:pipeline)可以将多个命令串联起来,每一个进程的 stdout 作为下一个进程的 stdin,在 shell 中使用频率很高。比如一个命令输出的内容很多,我们可以用管道加上 more 便可以分页阅读,或者使用 grep 进行过滤。

ps -ef | more
ps -ef | head
ps -ef | grep init

# 排序去重并统计行数
cat file | sort | uniq | wc-l

sort 排序、uniq 去重、wc -l 统计行数,这些都是 shell 常用的命令,用管道组合各种命令将更奇妙。

tee 命令

tee 经常与管道组合起来使用,可在 stdout 正常输出的同时另外保存一份到文件。例如将 ping 得到的结果保存到文件:

$  ping -c 3 www.baidu.com | tee ping.log
PING www.a.shifen.com (180.101.49.11) 56(84) bytes of data.
64 bytes from 180.101.49.11 (180.101.49.11): icmp_seq=1 ttl=52 time=17.1 ms
64 bytes from 180.101.49.11 (180.101.49.11): icmp_seq=2 ttl=52 time=19.4 ms
64 bytes from 180.101.49.11 (180.101.49.11): icmp_seq=3 ttl=52 time=16.7 ms

--- www.a.shifen.com ping statistics ---
3 packets transmitted, 3 received, 0% packet loss, time 5ms
rtt min/avg/max/mdev = 16.736/17.737/19.396/1.186 ms
$  
$  cat ping.log
PING www.a.shifen.com (180.101.49.11) 56(84) bytes of data.
64 bytes from 180.101.49.11 (180.101.49.11): icmp_seq=1 ttl=52 time=17.1 ms
...

tee 每次覆盖文件,tee -a 可附加文本到文件。

重定向和 sudo

正常的重定向无法配合 sudo 使用,解决的方法是通过 root 调用一个子 shell,在这个子 shell 里面进行重定向。例如:

$  sudo echo "some text" >> /root/foo.txt
bash: /root/foo.txt: 权限不够
$  
$  sudo bash -c 'echo "some text" >> /root/foo.txt'
$  sudo cat /root/foo.txt
some text

另外 tee 命令可以与 sudo 配合使用,例如:

$  echo "append" | sudo tee -a /root/foo.txt
append
$  sudo cat /root/foo.txt
some text
append
$  # tee 写入文件并将 stdout 屏蔽
$  echo "append again" | sudo tee -a /root/foo.txt > /dev/null

Warning

把文件覆盖了就等于删除了,所以这类操作要小心,追加写入使用 >>tee -a