Linux 重定向是指修改原来预设的一些东西,对原来系统命令的预设执行方式进行改变,比如说简单的我不想看到在显示器的输出而是希望输出到某一档案中就可以通过 Linux 重定向来进行这项工作。
Linux 预设输入是键盘,输出是显示器。你可以用重定向来改变这些设定。比如用 wc 命令的时候本来是要手动输入一篇文字来计算字元数的,用了重定 向后可以直接把一个已经写好的档案用 ‘<’ 指向这条命令,就直接可以统计这个档案的字元数等了。输出也是一样,你可以把萤幕输出重定向到一个档案 里,再到档案里去看结果。重定向操作符可以用来将命令输入和输出资料流从预设位置重定向到其他位置,其输入或输出资料流的位置称为控制代码; 常见的控制代码有三 种,当然控制代码可以自行扩充套件,一般的 OS 都提供类似的功能。控制代码 控制代码代号 控制代码描述
STDIN 0 键盘输入
STDOUT 1 输出资讯到提示符视窗
STDERR 2 输出错误资讯到提示符视窗
预设的 < 重定向输入操作符是 0,而预设的 > 重定向输出操作符是 1 。键入 < 或 > 操作符之后,必须指定资料的读写位置,可以是档名或其他现有的控制代码。
要指定重定向到现有控制代码,请使用与 & 字元,后面接要重定向的控制代码号 (即 & 控制代码号) 。
例如,下面的命令可以将控制代码 2(即 STDERR)重定向到控制代码 1(即 STDOUT):2>&1
下表列出了可用于重定向输入和输出资料流的操作符:
Linux 重定向操作符
功能描述
>
将命令输出写入档案或装置,而不是命令提示符或控制代码
<
从档案而不是从键盘或控制代码读入命令输入
>>
将命令输出新增到档案末尾而不删除档案中已有的资讯
>&
将一个控制代码的输出写入到另一个控制代码的输入中
<&
从一个控制代码读取输入并将其写入到另一个控制代码输出中
|
从一个命令中读取输出并将其写入另一个命令的输入中; 也称为管道操作符
现在我们回过头来看看上面的那条语句mysh > mylog.txt 2>&1 就可明白:
> mylog.txt意思是将标准输出重定向到mylog.txt,等价于mysh 1> mylog.txt;
2 >& 1 意思是将错误输出重定向到控制代码 1 标准输出; 综合起来就是 mysh 命令执行过程中产生的标准输出和错误输出都会被重定向到 mylog.txt 中;
重定向的功能十分强大,有兴趣的可以去尝试各种不同的组合,看看前后位置变下会有什么结果?
某些时候我们可能并不希望记录什么标准输出或者是错误输出,那可以用mysh >null 2>null或者mysh >/dev/null 2>/dev/null;
I/O 重定向详解
1 、 基本概念 (这是理解后面的知识的前提,请务必理解)
a 、 I/O 重定向通常与 FD 有关,shell 的 FD 通常为 10 个,即 0~9;
b 、 常用 FD 有 3 个,为 0(stdin,标准输入) 、 1(stdout,标准输出) 、 2(stderr,标准错误输出),预设与 keyboard 、 monitor 、 monitor 有关;
c 、 用 < 来改变读进的资料通道(stdin),使之从指定的档案读进;
d、 用 > 来改变送出的资料通道 (stdout, stderr),使之输出到指定的档案;
e 、 0 是 < 的预设值,因此 < 与 0<是一样的; 同理,> 与 1> 是一样的;
f 、 在 IO 重定向 中,stdout 与 stderr 的管道会先准备好,才会从 stdin 读进资料;
g 、 管道 “|”(pipe line): 上一个命令的 stdout 接到下一个命令的 stdin;
h 、 tee 命令是在不影响原本 I/O 的情况下,将 stdout 复制一份到档案去;
i 、 bash(ksh) 执行命令的过程:分析命令-变数求值-命令替代 (“和 $)-重定向-万用字元展开-确定路径-执行命令;
j 、 将 command group 置于 sub-shell 去执行,也称 nested sub-shell,它有一点非常重要的特性是:继承父 shell 的 Standard input, output, and error plus any other open file descriptors 。
k 、 exec 命令:常用来替代当前 shell 并重新启动一个 shell,换句话说,并没有启动子 shell 。使用这一命令时任何现有环境都将会被清除。 exec 在对档案描述符进行操作的时候,也只有在这时,exec 不会覆盖你当前的 shell 环境。
2 、 基本 IO
cmd > file 把 stdout 重定向到 file 档案中;
cmd >> file 把 stdout 重定向到 file 档案中 (追加);
cmd 1> fiel 把 stdout 重定向到 file 档案中;
cmd > file 2>&1 把 stdout 和 stderr 一起重定向到 file 档案中;
cmd 2> file 把 stderr 重定向到 file 档案中;
cmd 2>> file 把 stderr 重定向到 file 档案中 (追加);
cmd >> file 2>&1 把 stderr 和 stderr 一起重定向到 file 档案中 (追加);
cmd < file >file2 cmd 命令以 file 档案作为 stdin,以 file2 档案作为 stdout;
cat <>file 以读写的方式开启 file;
cmd < file cmd 命令以 file 档案作为 stdin;
cmd << delimiter Here document,从 stdin 中读入,直至遇到 delimiter 分界符。
3、 进阶IO
>&n 使用系统呼叫 dup (2) 复制档案描述符 n 并把结果用作标准输出;
<&n 标准输入复制自档案描述符 n;
<&- 关闭标准输入(键盘);
>&- 关闭标准输出;
n<&- 表示将 n 号输入关闭;
n>&- 表示将 n 号输出关闭;
上述所有形式都可以前导一个数字,此时建立的档案描述符由这个数字指定而不是预设的 0 或 1 。如:
… 2>file 执行一个命令并把错误输出 (档案描述符 2) 定向到 file 。
… 2>&1 执行一个命令并把它的标准输出和输出合并。 (严格的说是通过复制档案描述符 1 来建立档案描述符 2 ,但效果通常是合并了两个流。)
我们对 2>&1详细说明一下 :2>&1 也就是 FD2=FD1 ,这里并不是说FD2 的值 等于FD1的值,因为 > 是改变送出的资料通道,也就是说把 FD2 的 “资料输出通道” 改为 FD1 的 “资料输出通道” 。如果仅仅这样,这个改变好像没有什么作用,因为 FD2 的预设输出和 FD1 的预设输出本来都是 monitor,一样的! 但是,当 FD1 是其他档案,甚至是其他 FD 时,这个就具有特殊的用途了。请大家务必理解这一点。
exec 0exec 1>outfilename # 开启档案 outfilename 作为 stdout 。
exec 2>errfilename # 开启档案 errfilename 作为 stderr 。
exec 0<&- # 关闭 FD0。
exec 1>&- # 关闭 FD1 。
exec 5>&- # 关闭 FD5 。
这样就完成了 Linux 重定向的学习。
原文出自【位元网】,转载请保留原文连结:http://soft.chinabyte.com/os/451/11757951.shtml
1 重定向
1.1 重定向符号
> 输出重定向到一个档案或装置 覆盖原来的档案
> ! 输出重定向到一个档案或装置 强制覆盖原来的档案
>> 输出重定向到一个档案或装置 追加原来的档案
< 输入重定向到一个程式
1.2标准错误重定向符号
2> 将一个标准错误输出重定向到一个档案或装置 覆盖原来的档案 b-shell
2>> 将一个标准错误输出重定向到一个档案或装置 追加到原来的档案
2>&1 将一个标准错误输出重定向到标准输出 注释:1 可能就是代表 标准输出
>& 将一个标准错误输出重定向到一个档案或装置 覆盖原来的档案 c-shell
|& 将一个标准错误 管道 输送 到另一个命令作为输入
1.3 命令重导向示例
在 bash 命令执行的过程中,主要有三种输出入的状况,分别是:
1. 标准输入;程式码为 0 ;或称为 stdin ;使用的方式为 <
2. 标准输出:程式码为 1 ;或称为 stdout;使用的方式为 1>
3. 错误输出:程式码为 2 ;或称为 stderr;使用的方式为 2>
[test @test test]# ls -al > list.txt
将显示的结果输出到 list.txt 档案中,若该档案以存在则予以取代!
[test @test test]# ls -al >> list.txt
将显示的结果累加到 list.txt 档案中,该档案为累加的,旧资料保留!
[test @test test]# ls -al 1> list.txt 2> list.err
将显示的资料,正确的输出到 list.txt 错误的资料输出到 list.err
[test @test test]# ls -al 1> list.txt 2> &1
将显示的资料,不论正确或错误均输出到 list.txt 当中!错误与正确档案输出到同一个档案中,则必须以上面的方法来写!不能写成其它格式!
[test @test test]# ls -al 1> list.txt 2> /dev/null
将显示的资料,正确的输出到 list.txt 错误的资料则予以丢弃! /dev/null ,可以说成是黑洞装置。为空,即不储存。
1.4 为何要使用命令输出重导向
• 当萤幕输出的资讯很重要,而且我们需要将他存下来的时候;
• 背景执行中的程式,不希望他干扰萤幕正常的输出结果时;
• 一些系统的例行命令(例如写在 /etc/crontab 中的档案)的执行结果,希望他可以存下来时;
• 一些执行命令,我们已经知道他可能的错误讯息,所以想以『2> /dev/null』将他丢掉时;
• 错误讯息与正确讯息需要分别输出时。
2 管线命令 ( pipe )
就如同前面所说的, bash 命令执行的时候有输出的资料会出现,那么如果这群资料必需要经过几道手续之后才能得到我们所想要的格式,应该如何来设定?这就牵涉到管线命令的问题了(pipe),管线命令使用的是『|』。
例子:简单的管线命令
假设我们要读取 last 这个指令中,那个 root 登入的『次数』应该怎么作?
那么我所进行的步骤是:
1. 执行 last ,将所有这个月的所有人登入资料取出来;
2. 使用 grep 将上面的输出资料(stdout)当中的 root 撷取出来,其它的不要;
3. 使用 wc 这个可以计算行数的指令将上一步的资料计算行数!
由于 last 的输出是一行代表一次登入,所以只要计算几行就代表登入几次的意思,经由上面三个步骤,将 last 资料逐步的筛选,就可以得到我们的资料了!整个命令可以写成如下:
[test @test bin]# last | grep root | wc -l
这个管线命令『|』仅能处理经由前面一个指令传来的正确资讯,也就是 standard output ( STDOUT ) 的资讯,对于 stdandard error 并没有直接处理的能力。
2.1 基本的管线命令指令介绍
• cut
语法:[root @test /root ]# cut -d “分隔字元” [-cf] fields
引数说明:
-d :后面接的是用来分隔的字元,预设是『空格符』
-c :后面接的是『第几个字元』
-f :后面接的是第几个区块?
范例:[root @test /root]# cat /etc/passwd | cut -d “:” -f 1
将 passwd 这个档案里面,每一行里头的 : 用来作为分隔号,而列出第一个区块!也就是姓名所在啦!
[root @test /root]# last | cut -c1-20
将 last 之后的资料,每一行的 1-20 个字元取出来!
• sort
语法:[root @test /root ]# sort [-t 分隔符] [(+起始)(-结束)] [-nru]
引数说明:
-t 分隔符:使用分隔符来隔开不同区间,预设是 tab
+start -end:由第 start 区间排序到 end 区间
-n :使用『纯数字』排序(否则就会以文字型态来排序)
-r :反向排序
-u :相同出现的一行,只列出一次!
范例:
[root @test /root]# cat /etc/passwd | sort 将列出来的个人账号排序!
[root @test /root]# cat /etc/passwd | sort -t: +2n 将个人账号中,以使用者 ID 来排序(以 : 来分隔,第三个为 ID ,但第一个代号为 0 之故)
[root @test /root]# cat /etc/passwd | sort -t: +2nr 反相排序啰!
• wc
语法:[root @test /root ]# wc [-lmw]
引数说明:
-l :多少行
-m :多少字元
-w :多少字
范例:
[root @test /root]# cat /etc/passwd | wc -l 这个档案里头有多少行?
[root @test /root]# cat /etc/passwd | wc -w 这个档案里头有多少字!?
• uniq 这个指令用来将『重复的行删除掉只显示一个』
语法:[root @test /root ]# uniq
范例:
[root @test /root]# last | cut -d” ” -f1 | sort | uniq
• tee 命令重定向到档案的同时将资料显示在萤幕上
语法:[root @test /root ]# last | tee last.list | cut -d ” ” -f1
范例:
[root @test /root]# last | tee last.list | cut -d ” ” -f1
• tr
语法:[root @test /root ]# tr [-ds] SET1
引数说明:
-d :删除 SET1 这个字串
-s :取代掉重复的字元!
范例:
[root @test /root]# last | tr ‘[a-z]‘ ‘[A-Z]‘ <==将小写改成大写 [root @test /root]# cat /etc/passwd | tr -d : <== : 这个符号在 /etc/passwd 中不见了! [root @test /root]# cat /home/test/dostxt | tr -d ‘r’ > dostxt-noM
• split
语法:[root @test /root ]# split [-bl] 输入档案 输出档案前导字元
引数说明:
-b :以档案 size 来分
-l :以行数来分
范例:
[root @test /root]# split -l 5 /etc/passwd test <==会产生 testaa, testab, testac… 等等的档案 说明:在 Linux 底下就简单的多了!你要将档案分割的话,那么就使用 -b size 来将一个分割的档案限制其大小,如果是行数的话,那么就使用 -l line 来分割! 管线命令在 bash 的连续的处理程式中是相当重要的!另外,在 log file 的分析当中也是相当重要的一环。 管道输送到一个命令的标准输入可以使用标准输入引数”-“ 进行更仔细的控制. 如 cat 命令的示例 原文连结:http://outofmemory.cn/wr/?u=http%3A%2F%2Fopkeep.com%2F