第 3 章目录:
3.1 shell 语法
3.2 shell 命令
3.3 shell 函数
3.4 shell 参数
3.5 shell 展开
3.6 重定向
3.7 命令执行
3.8 shell 脚本
本文件内容为 "3.6 重定向"
对于一条命令而言,我们关心的是命令的输入,以及命令的输出。命令的输出有两种,一是标准输出,一是命令的错误信息输出。默认情况下,命令的标准输出写入到“标准输出文件”,其文件描述符为1。标准输出文件是一个特殊的文件,其中的内容是直接展示到屏幕上的。命令的错误输出默认也是写入“标准输出文件”。命令的标准输入,默认从“标准输入文件”中读取内容,其文件描述符为 0。标准输入文件直接与我们的键盘输入设备关联。
举例来讲,比如我们执行 ls 命令,在屏幕上直接会显示当前目录的内容:
[root@vm2 zabbix]# ls
zabbix_agentd.log
这就是因为 ls 命令的标准输出是写入到“标准输出文件”中,因为“标准输出文件”很特殊,它的内容会直接展示到屏幕上,所以我们执行 ls 命令才能直接看到类似上面的输出。
对于命令的标准输入来讲,比如,执行 cat - 命令,cat 命令会直接显示指定文件的内容(把指定文件内容写到“标准输出文件”),cat - 表示显示“标准输入文件”的内容,这时就会等待我们从键盘输入,这里随意输入一个字符串,按回车之后便显示出来了:
[guli@gulideMacBook-Air ~]$ cat -
alskdf
alskdf
而我们要做重定向,其实就是改变默认的设定,将命令的输入和输出与其他文件关联。
如果对命令做了重定向,在命令执行之前,命令的输入输出可被重定向。
重定向也可用于为当前的 shell 环境打开文件或关闭文件。
重定向符号的位置:重定向符号可出现在一个简单命令的之前,之中和之后。
重定向按照出现的顺序,从左至右执行。
在重定向符号之前可跟一个文件描述符,比如 2>;也可跟 {VARNAME}。
如果是 {VARNAME},对于除了 '>&-' 和 '<$-' 之外的所有重定向符号,一个大于 10 的文件描述符将被分配给 {VARNAME}。
对于 '>&-' 和 '<$-',VARNAME 的值定义了一个文件描述符,其关联的文件将被关闭。
在下面的描述中,如果没有显式地写出文件描述符,那么:
第一个重定向符号为 '<' 时,表示 标准输入 文件(文件描述符为 0)。
第一个重定向符号为 '>' 时,表示 标准输出 文件(文件描述符为 1)。
跟在重定向符号之后的单词,除非另有说明,可进行如下展开:
花括号展开,波浪线展开,参数展开,命令替换,算术展开,引用取消,文件名展开 以及 单词分割。
如果展开为多个单词,bash 会报告一个错误。
注意,重定向的顺序是很重要的。比如:
ls > DIRLIST 2>&1
表示先对 “标准输出”(文件描述符为1)做重定向,文件描述符1 被关联到 DIRLIST 文件。
然后对 “标准错误输出”(文件描述符为 2)做重定向,这里使用了一个引用 &1,表示与文件描述符1关联的文件,因为已经做了重定向,其值为 DIRLIST,于是标准错误输出被重定向到了 DIRLIST 文件。
最终效果是同时对将 标准输出 和 标准错误输出(文件描述符为 2) 导入到 DIRLIST 文件。
ls 2>&1 > DIRLIST
对比上一条,这条命令只把标准输出导入文件 DIRLIST。
2>&1
表示对标准错误输出做重定向,也就是准备改变文件描述符 2 所关联的文件。这里重定向到文件描述符 1 关联的文件,关联的文件仍然是默认的 “标准输出文件”。
>
表示对标准输出做重定向,也就是改变文件描述符 1 所关联的文件,这里被重定向到了 DIRLIST。
bash 在做重定向时,对一些文件名进行特殊对待:
'/dev/fd/FD'
如果 FD 是有效的整数,文件描述符 FD 被复制。
'/dev/stdin'
文件描述符 0 被复制。
'/dev/stdout'
文件描述符 1 被复制。
'/dev/stderr'
文件描述符 2 被复制。
'/dev/tcp/HOST/PORT'
如果 HOST 是有效的主机名,或是 IP 地址,而且 PORT 是一个整数的端口数字,或是 服务名字,那么 bash 会尝试打开一个 TCP 连接到对应的 socket。
'/dev/udp/HOST/PORT'
如果 HOST 是有效的主机名,或是 IP 地址,而且 PORT 是一个整数的端口数字,或是 服务名字,那么 bash 会尝试打开一个 UDP 连接到对应的 socket。
如果打开文件失败,或者创建文件失败,将导致重定向失败。
大于 9 的文件描述符应谨慎使用,因为 shell 内部可能会用到。
3.6.1 输入重定向
以 WORD 的展开结果为文件名。输入重定向使得以该文件名命名的文件
与文件描述符 'N' 关联,且文件被打开并等待读取。
如果 'N' 未指定,文件被关联到标准输入。
[N]
3.6.2 输出重定向
以 WORD 的展开结果为文件名。输入重定向使得以该文件名命名的文件与文件描述符 'N' 关联,且文件被打开并等待写入。如果 'N' 未指定,文件被关联到标准输出。
如果文件不存在,将被创建。如果文件存在,文件内容被清空(truncated)。
[N]>[|]WORD
如果重定向符号是 '>',且 'noclobber' 选项被 set 命令开启,如果文件已存在,而且文件是普通文件,重定向会失败。
如果重定向符号是 '>|',或者是 '>' 且 'noclobber' 未开启,那么即使文件已经存在,也会尝试进行重定向。
3.6.3 追加输出重定向
追加输出重定向,会将输出追加到文件已有的内容之后。如果文件不存在,将被创建。
[N]>>WORD
3.6.4 重定向标准输出和标准错误输出
这个构造允许 标准输出 和 标准错误输出 同时重定向到文件。
有两种形式:
&>WORD
>&WORD
以上两者中,第一种更受欢迎 &>WORD。
在语义上,它们与此相同:
>WORD 2>&1
>WORD: 即 1>WORD,将WORD文件跟 “文件描述符1” 关联,等待写入。
表示标准输出重定向到WORD文件。
2>&1: 表示将 “文件描述符2” 跟 “文件描述符1” 所关联的文件做关联,
等待写入。
3.6.5 追加重定向标准输出和标准错误输出
这个构造允许 标准输出 和 标准错误输出 同时重定向到文件。
其形式为:
&>>WORD
在语义上,它与此相同:
>>WORD 2>>&1
3.6.6 Here Document
这种重定向从当前 source 读取输入,直到出现只包含 WORD 的行出现(其后不跟任何 blank 字符)。此前读取的所有行,将被当做一个命令的标准输入来使用。
<<[-]WORD
HERE-DOCUMENT
DELIMITER
对 WORD 不会进行 参数展开,命令替换,算术展开 和 文件名展开。
如果 WORD 中包含被引用的字符,DELIMITER 等于引用删除后的值,here-document 中的所有行不做展开。
如果 WORD 未被引用,here-document 中的所有行可进行 参数展开,命令替换 和 算术展开。'\newline' 被忽略,而且 '' 必须用来引用 '','$' 以及 '`'。
如果重定向符号是 '<<-',所有 leading tab 字符将从输入行中删除,包括含有 DELIMITER 的行。这允许在 shell script 中的 here-document 以自然的方式进行缩进。
3.6.7 Here String
这是 here-document 的变体:
<<< WORD
WORD 被展开,然后提供给一个命令的标准输入。
3.6.8 文件描述符的复制
输入文件描述符的复制:
[N]<&WORD
这个重定向运算符被用来对输入文件描述符做复制。
如果 WORD 展开为一个或多个数字,描述符 'N' 关联的文件被做成展开数字(作为文件描述符)所关联文件的一个拷贝。
如果 WORD 不是一个以读方式打开的文件描述符,会导致重定向报错。如果 WORD 等于 '-',描述符 'N' 被关闭。
如果 'N' 没有指定,则使用 标准输入(文件描述符为 0)。
输出文件描述符的复制:
[N]>&WORD
这个重定向运算符被用来对输出文件描述符做复制。如果 WORD 展开为一个或多个数字,描述符 'N' 关联的文件被做成展开数字(作为文件描述符)所关联文件的一个拷贝。
如果 WORD 不是一个以写方式打开的文件描述符,会导致重定向报错。
如果 'N' 没有指定,默认为1.
如果 'N' 没有指定,且 WORD 没有展开为数字,那么标准输出和标准错误输出会被重定向(如前面的小节所述)。
3.6.9 文件描述符移动
[N]<&DIGIT-
将文件描述符 DIGIT 移动到 N,如果 N 没指定,则移动到 0。DIGIT 被复制到 N 后,DIGIT 描述符被关闭。
[N]>&DIGIT-
将文件描述符 DIGIT 移动到 N,如果 N 没指定,则移动到 1。
3.6.10 以读写方式打开文件描述符
[N]<>WORD
对 WORD 进行展开,以展开结果为文件名,将该文件与描述符 'N' 关联,以读写方式打开。如果 N 没指定,使用 0 描述符。如果文件不存在则创建文件。