您的当前位置:首页正文

Shell零基础学习

2024-07-29 来源:易榕旅网
Shell脚本

1.Shell 是一个用 C 语言编写的程序,它是用户使用 Linux 的桥梁。 Shell 既是一种命令语言,又是一种程序设计语言。 Shell 是指一种应用程序,这个应用程序提供了一个界面,用户通过这个界面访问操作系统内核的服务。

Shell 脚本(shell script),是一种为 shell 编写的脚本程序。

2.Linux 的 Shell 种类众多,常见的有:

    

Bourne Shell(/usr/bin/sh或/bin/sh) Bourne Again Shell(/bin/bash) C Shell(/usr/bin/csh) K Shell(/usr/bin/ksh) Shell for Root(/sbin/sh)

由于易用和免费,Bash 在日常工作中被广泛使用。同时,Bash 也是大多数Linux 系统默认的 Shell。 在一般情况下,人们并不区分 Bourne Shell 和 Bourne Again Shell,所以,像 #!/bin/sh,它同样也可以改为 #!/bin/bash。

#! 告诉系统其后路径所指定的程序即是解释此脚本文件的 Shell 程序。

3.echo 命令用于向窗口输出文本。

Shell变量

1.定义变量时,变量名不加美元符号($,PHP语言中变量需要) 变量名和等号之间不能有空格 变量名的命名须遵循如下规则:

  

命名只能使用英文字母,数字和下划线,首个字符不能以数字开头。 中间不能有空格,可以使用下划线(_)。 不能使用标点符号。

 不能使用bash里的关键字(可用help命令查看保留关键字)。

2.使用变量

使用一个定义过的变量,只要在变量名前面加美元符号即可,例如:

your_name=\"qinjx\" echo $your_name echo ${your_name}

变量名外面的花括号是可选的,加不加都行,加花括号是为了帮助解释器识别变量的边界; 推荐给所有变量加上花括号,这是个好的编程习惯。

3.只读变量

使用 readonly 命令可以将变量定义为只读变量,只读变量的值不能被改变。 语法:readonly your_name

4.删除变量

使用 unset 命令可以删除变量。 语法:unset variable_name 变量被删除后不能再次使用。 unset 命令不能删除只读变量。

5.变量类型

运行shell时,会同时存在三种变量:

 1) 局部变量 局部变量在脚本或命令中定义,仅在当前shell实例中有效,其他shell启动的程序不能访问局部变量。

 2) 环境变量 所有的程序,包括shell启动的程序,都能访问环境变量,有些程序需要环境变量来保证其正常运行。

注:必要的时候shell脚本也可以定义环境变量。

 3) shell变量 shell变量是由shell程序设置的特殊变量。

注:shell变量中有一部分是环境变量,有一部分是局部变量,这些变量保证了

shell的正常运行。

Shell字符串

字符串是shell编程中最常用最有用的数据类型(除了数字和字符串,也没啥其它类型好用了),字符串可以用单引号,也可以用双引号,也可以不用引号。单双引号的区别跟PHP类似。

1.单引号

str='this is a string' 单引号字符串的限制:

 

单引号里的任何字符都会原样输出,单引号字符串中的变量是无效的; 单引号字串中不能出现单引号(对单引号使用转义符后也不行)。

2.双引号

your_name='qinjx' str=\"Hello, I know you are \\\"$your_name\\\"! \\n\" 双引号的优点:

 

双引号里可以有变量 双引号里可以出现转义字符

3.拼接字符串

your_name=\"qinjx\" greeting=\"hello, \"$your_name\" !\" greeting_1=\"hello, ${your_name} !\"echo $greeting $greeting_1

4.获取字符串长度

string=\"abcd\" echo ${#string}#输出 4

5.提取子字符串

从字符串第 2 个字符开始截取 4 个字符:

string=\"runoob is a great site\"

echo ${string:1:4}# 输出 unoo

6.查找子字符串

查找字符 \"i 或 s\" 的位置:

string=\"runoob is a great company\"

echo `expr index \"$string\" is`# 输出 8

注意: 以上脚本中 \"`\" 是反引号,而不是单引号 \"'\",不要看错了哦。

Shell数组

bash支持一维数组(不支持多维数组),并且没有限定数组的大小。 数组元素的下标由0开始编号。获取数组中的元素要利用下标,下标可以是整数或算术表达式,其值应大于或等于0。

1.定义数组

在Shell中,用括号来表示数组,数组元素用\"空格\"符号分割开。 定义数组的一般形式为:

数组名=(值1值2...值n) 例子:

array_name=(value0 value1 value2 value3) 还可以单独定义数组的各个分量:

array_name[0]=value0 array_name[1]=value1

array_name[n]=valuen

可以不使用连续的下标,而且下标的范围没有限制。

2.读取数组

读取数组元素值的一般格式是:

${数组名[下标]}

使用 @ 或 * 符号可以获取数组中的所有元素,例如:

echo ${array_name[@]} echo ${array_name[*]}

3.获取数组的长度

获取数组长度的方法与获取字符串长度的方法相同,例如:

# 取得数组元素的个数 length=${#array_name[@]}# 或者

length=${#array_name[*]}# 取得数组单个元素的长度 lengthn=${#array_name[n]}

Shell注释

以\"#\"开头的行就是注释,会被解释器忽略。 sh里没有多行注释,只能每一行加一个#号。

如果在开发过程中,遇到大段的代码需要临时注释起来,过一会儿又取消注释,怎么办呢? 每一行加个#符号太费力了,可以把这一段要注释的代码用一对花括号括起来,定义成一个函数,没有地方调用这个函数,这块代码就不会执行,达到了和注释一样的效果。

Shell传递参数

在执行 Shell 脚本时,向脚本传递参数,脚本内获取参数的格式为:$n。n 代表一个数字,1 为执行脚本的第一个参数,2 为执行脚本的第二个参数,以此类推……($0 为执行的文件名)

用来处理参数的几个特殊字符: 参数处理 说明 $# $* 传递到脚本的参数个数 以一个单字符串显示所有向脚本传递的参数。(将传递的参数作为一个字符串) 如\"$*\"用「\"」括起来的情况、以\"$1 $2 … $n\"的形式输出所有参数。 $$ $! $@ 脚本运行的当前进程ID号 后台运行的最后一个进程的ID号 与$*相同,但是使用时加引号,并在引号中返回每个参数。 如\"$@\"用「\"」括起来的情况、以\"$1\" \"$2\" … \"$n\" 的形式输出所有参数。 $- $? 显示Shell使用的当前选项,与set命令功能相同。 显示最后命令的退出状态。0表示没有错误,其他任何值表明有错误。 $* 与 $@ 区别:

 

相同点:都是引用所有参数。

不同点:只有在双引号中体现出来。假设在脚本运行时写了三个参数 1、2、3,,则 \" * \" 等价于 \"1 2 3\"(传递了一个参数),而 \"@\" 等价于 \"1\" \"2\" \"3\"(传递了三个参数)。

Shell基本运算符

原生bash不支持简单的数学运算,但是可以通过其他命令来实现,例如 awk 和 expr,expr 最常用。

expr 是一款表达式计算工具,使用它能完成表达式的求值操作。 例如,两个数相加(注意使用的是反引号 ` 而不是单引号 '):

#!/bin/bash val=`expr 2 + 2` echo \"两数之和为 : $val\"

两点注意:

 表达式和运算符之间要有空格,例如 2+2 是不对的,必须写成 2 + 2,这与我们熟悉的大多数编程语言不一样。

 完整的表达式要被 ` ` 包含,注意这个字符不是常用的单引号,在 Esc 键下边。

1.算数运算符

下表列出了常用的算术运算符,假定变量 a 为 10,变量 b 为 20: 运算说明 符 + - * / % = == 加法 减法 乘法 除法 取余 赋值 相等。用于比较两个数字,相同则返回 true。 `expr $a + $b` 结果为 30。 `expr $a - $b` 结果为 -10。 `expr $a \\* $b` 结果为 200。 `expr $b / $a` 结果为 2。 `expr $b % $a` 结果为 0。 a=$b 将把变量 b 的值赋给 a。 [ $a == $b ] 返回 false。 [ $a != $b ] 返回 true。 举例 != 不相等。用于比较两个数字,不相同则返回 true。 注意:条件表达式要放在方括号之间,并且要有空格,例如: [$a==$b] 是错误的,必须

写成 [ $a == $b ]。 注意:

 乘号(*)前边必须加反斜杠(\\)才能实现乘法运算;

 

if...then...fi 是条件语句,后续将会讲解。

在 MAC 中 shell 的 expr 语法是:$((表达式)),此处表达式中的 \"*\" 不需要转义符号 \"\\\" 。

if[ $a == $b ]then echo \"a 等于 b\"fiif[ $a != $b ]then echo \"a 不等于 b\"fi

2.关系运算符

关系运算符只支持数字,不支持字符串,除非字符串的值是数字。 下表列出了常用的关系运算符,假定变量 a 为 10,变量 b 为 20: 运算说明 符 -eq -ne 检测两个数是否相等,相等返回 true。 检测两个数是否不相等,不相等返回 true。 [ $a -eq $b ] 返回 false。 [ $a -ne $b ] 返回 true。 举例 -gt 检测左边的数是否大于右边的,如果是,则返回 true。 [ $a -gt $b ] 返回 false。 -lt 检测左边的数是否小于右边的,如果是,则返回 true。 [ $a -lt $b ] 返回 true。 检测左边的数是否大于等于右边的,如果是,则返回 -ge true。 检测左边的数是否小于等于右边的,如果是,则返回 -le true。 3.布尔运算符

下表列出了常用的布尔运算符,假定变量 a 为 10,变量 b 为 20:

[ $a -ge $b ] 返回 false。 [ $a -le $b ] 返回 true。 运算说明 符 ! 非运算,表达式为 true 则返回 false,否则返回 [ ! false ] 返回 true。 举例 true。 [ $a -lt 20 -o $b -gt 100 ] 返-o 或运算,有一个表达式为 true 则返回 true。 回 true。 [ $a -lt 20 -a $b -gt 100 ] 返-a 与运算,两个表达式都为 true 才返回 true。 回 false。

4.逻辑运算符

Shell 的逻辑运算符,假定变量 a 为 10,变量 b 为 20:

运算符 说明 举例 && 逻辑的 AND [[ $a -lt 100 && $b -gt 100 ]] 返回 false || 逻辑的 OR [[ $a -lt 100 || $b -gt 100 ]] 返回 true

5.字符串运算符

下表列出了常用的字符串运算符,假定变量 a 为 \"abc\",变量 b 为 \"efg\":

运算符 = 说明 举例 检测两个字符串是否相等,相等返回 true。 [ $a = $b ] 返回 false。 != 检测两个字符串是否相等,不相等返回 true。 [ $a != $b ] 返回 true。 -z 检测字符串长度是否为0,为0返回 true。 [ -z $a ] 返回 false。 -n 检测字符串长度是否为0,不为0返回 true。 [ -n \"$a\" ] 返回 true。 str 检测字符串是否为空,不为空返回 true。 [ $a ] 返回 true。

6.文件测试运算符

文件测试运算符用于检测 Unix 文件的各种属性。 属性检测描述如下:

操作说明 符 举例 -b 检测文件是否是块设备文件,如果是,则返回 true。 file -c 检测文件是否是字符设备文件,如果是,则返回 true。 file -d 检测文件是否是目录,如果是,则返回 true。 file 检测文件是否是普通文件(既不是目录,也不是设备文件),如-f file 果是,则返回 true。 -g 检测文件是否设置了 SGID 位,如果是,则返回 true。 file -k 检测文件是否设置了粘着位(Sticky Bit),如果是,则返回 true。 file -p 检测文件是否是有名管道,如果是,则返回 true。 file -u 检测文件是否设置了 SUID 位,如果是,则返回 true。 file [ -b $file ] 返回 false。 [ -c $file ] 返回 false。 [ -d $file ] 返回 false。 [ -f $file ] 返回 true。 [ -g $file ] 返回 false。 [ -k $file ] 返回 false。 [ -p $file ] 返回 false。 [ -u $file ] 返回 false。 [ -r $file ] 返-r file 检测文件是否可读,如果是,则返回 true。 回 true。 -w 检测文件是否可写,如果是,则返回 true。 file -x 检测文件是否可执行,如果是,则返回 true。 [ -w $file ] 返回 true。 [ -x $file ] 返file -s 检测文件是否为空(文件大小是否大于0),不为空返回 true。 file -e 检测文件(包括目录)是否存在,如果是,则返回 true。 file

回 true。 [ -s $file ] 返回 true。 [ -e $file ] 返回 true。 Shell echo命令

1.显示转义字符

echo \"\\\"It is a test\\\"\"

输出结果:

\"It is a test\"

2.显示变量

read 命令从标准输入中读取一行,并把输入行的每个字段的值指定给 shell 变量

#!/bin/sh read name echo \"$name It is a test\"

输出结果: OK #标准输入

OK Itis a test #输出

3.显示换行

echo -e \"OK! \\n\"# -e 开启转义 echo \"It it a test\"

输出结果:

OK!It it a test

4.显示不换行

echo -e \"OK! \\c\"# -e 开启转义 \\c 不换行 echo \"It is a test\"

输出结果:

OK!Itis a test

5.显示结果定向至文件

echo \"It is a test\"> myfile

6.原样输出字符串,不进行转义或取变量(用单引号)

echo '$name\\\"'

输出结果: $name\\\"

7.显示命令执行结果

echo `date`

输出结果:

ThuJul2410:08:46 CST 2018

注意: 这里使用的是反引号 `, 而不是单引号 '。 结果将显示当前日期

Shell printf命令

printf 命令的语法:

printf format-string[arguments...]

参数说明:

 

format-string: 为格式控制字符串 arguments: 为参数列表。

接下来,我来用一个脚本来体现printf的强大功能:

#!/bin/bash# author:菜鸟教程# url:www.runoob.com printf \"%-10s %-8s %-4s\\n\"姓名性别体重kg printf \"%-10s %-8s %-4.2f\\n\"郭靖男66.1234 printf \"%-10s %-8s %-4.2f\\n\"杨过男48.6543 printf \"%-10s %-8s %-4.2f\\n\"郭芙女47.9876

执行脚本,输出结果如下所示:

姓名性别体重kg 郭靖男66.12杨过男48.65郭芙女47.99

%s %c %d %f都是格式替代符

%-10s 指一个宽度为10个字符(-表示左对齐,没有则表示右对齐),任何字符都会被显示在10个字符宽的字符内,如果不足则自动以空格填充,超过也会将内容全部显示出来。 %-4.2f 指格式化为小数,其中.2指保留2位小数。

%d %s %c %f 格式替代符详解:

d: Decimal 十进制整数 -- 对应位置参数必须是十进制整数,否则报错! s: String 字符串 -- 对应位置参数必须是字符串或者字符型,否则报错! c: Char 字符 -- 对应位置参数必须是字符串或者字符型,否则报错! f: Float 浮点 -- 对应位置参数必须是数字型,否则报错!

printf的转义序列

列 \\a \\b 说明 警告字符,通常为ASCII的BEL字符 后退 抑制(不显示)输出结果中任何结尾的换行字符(只在%b格式指示符控制下的参数\\c 字符串中有效),而且,任何留在参数里的字符、任何接下来的参数以及任何留在格式字符串中的字符,都被忽略 \\f \\n \\r \ \\v \\\\ \\ddd \\0ddd

换页(formfeed) 换行 回车(Carriage return) 水平制表符 垂直制表符 一个字面上的反斜杠字符 表示1到3位数八进制值的字符。仅在格式字符串中有效 表示1到3位的八进制值字符 Shell test命令

1.数值测试 参数 说明 -eq 等于则为真 -ne 不等于则为真 -gt 大于则为真 -ge 大于等于则为真 -lt 小于则为真 -le 小于等于则为真

num1=100 num2=100if test $[num1]-eq $[num2]then echo '两个数相等!'else echo '两个数不相等!'fi

输出结果:

两个数相等!

代码中的 [] 执行基本的算数运算,如: a=5 b=6 result=$[a+b]# 注意等号两边不能有空格

echo \"result 为: $result\"

结果为:

result 为:11

2.字符串测试 参数 = != 说明 等于则为真 不相等则为真 -z 字符串 字符串的长度为零则为真 -n 字符串 字符串的长度不为零则为真

num1=\"ru1noob\" num2=\"runoob\"if test $num1 = $num2 then echo '两个字符串相等!'else echo '两个字符串不相等!'fi

输出结果:

两个字符串不相等!

3.文件测试 参数 -e 文件名 -r 文件名 说明 如果文件存在则为真 如果文件存在且可读则为真 -w 文件名 -x 文件名 如果文件存在且可写则为真 如果文件存在且可执行则为真 -s 文件名 如果文件存在且至少有一个字符则为真 -d 文件名 -f 文件名 如果文件存在且为目录则为真 如果文件存在且为普通文件则为真 -c 文件名 如果文件存在且为字符型特殊文件则为真 -b 文件名 如果文件存在且为块特殊文件则为真

cd /bin if test -e ./bash then echo '文件已存在!'else echo '文件不存在!'fi

输出结果:

文件已存在!

另外,Shell还提供了与( -a )、或( -o )、非( ! )三个逻辑操作符用于将测试条件连接起来,其优先级为:\"!\"最高,\"-a\"次之,\"-o\"最低。例如:

cd /bin if test -e ./notFile -o -e ./bash then echo '至少有一个文件存在!'else echo '两个文件都不存在'fi

输出结果:

至少有一个文件存在!

Shell 流程控制

1.if

if 语句语法格式:

if condition then command1 command2 ... commandN fi

2.if else

if else 语法格式:

if condition then command1 command2 ... commandN else command fi

3.if else-if else

if else-if else 语法格式:

if condition1 then command1 elif condition2 then command2 else commandN fi

4.for循环

for循环一般格式为:

forvarin item1 item2 ... itemN do command1 command2 ... commandN done

写成一行:

forvarin item1 item2 ... itemN;do command1; command2…done;

当变量值在列表里,for循环即执行一次所有命令,使用变量名获取列表中的当前取值。命令可为任何有效的shell命令和语句。in列表可以包含替换、字符串和文件名。 in列表是可选的,如果不用它,for循环使用命令行的位置参数。

5.while语句

while循环用于不断执行一系列命令,也用于从输入文件中读取数据;命令通常为测试条件。其格式为:

while condition do command done

以下是一个基本的while循环,测试条件是:如果int小于等于5,那么条件返回真。int从0开始,每次循环处理时,int加1。运行上述脚本,返回数字1到5,然后终止。

#!/bin/shint=1while(( $int<=5))do echo $int let\"int++\"done

运行脚本,输出:

12345

while循环可用于读取键盘信息。下面的例子中,输入信息被设置为变量FILM,按结束循环。

echo '按下退出' echo -n '输入你最喜欢的网站名: 'while read FILM do echo \"是的!$FILM 是一个好网站\"done

运行脚本,输出类似下面:

按下退出输入你最喜欢的网站名:菜鸟教程是的!菜鸟教程是一个好网站

6.无限循环

无限循环语法格式:

while:do command done

或者

whiletruedo command done

或者

for((;;))

7.until循环

until 循环执行一系列命令直至条件为 true 时停止。 until 循环与 while 循环在处理方式上刚好相反。

一般 while 循环优于 until 循环,但在某些时候—也只是极少数情况下,until 循环更加有用。 until 语法格式:

until condition do command done

condition 一般为条件表达式,如果返回值为 false,则继续执行循环体内的语句,否则跳出循环。

以下实例我们使用 until 命令来输出 0 ~ 9 的数字:

#!/bin/bash a=0until[! $a -lt 10]do echo $a a=`expr $a + 1`done

输出结果为:

0123456789

8.case

Shell case语句为多选择语句。可以用case语句匹配一个值与一个模式,如果匹配成功,执行相匹配的命令。case语句格式如下:

case值in模式1) command1 command2 ... commandN ;;模式2) command1 command2 ...commandN ;;esac

case工作方式如上所示。取值后面必须为单词in,每一模式必须以右括号结束。取值可以为变量或常数。匹配发现取值符合某一模式后,其间所有命令开始执行直至 ;;。

取值将检测匹配的每一个模式。一旦模式匹配,则执行完匹配模式相应命令后不再继续其他模式。如果无一匹配模式,使用星号 * 捕获该值,再执行后面的命令。 下面的脚本提示输入1到4,与每一种模式进行匹配:

echo '输入 1 到 4 之间的数字:' echo '你输入的数字为:' read aNum case $aNum in1) echo '你选择了 1';;2) echo '你选择了 2';;3) echo '你选择了 3';;4) echo '你选择了 4';;*) echo '你没有输入 1 到 4 之间的数字';;esac

输入不同的内容,会有不同的结果,例如:

输入1到4之间的数字:你输入的数字为:3你选择了3

9.跳出循环

在循环过程中,有时候需要在未达到循环结束条件时强制跳出循环,Shell使用两个命令来实现该功能:break和continue。

break命令

break命令允许跳出所有循环(终止执行后面的所有循环)。

下面的例子中,脚本进入死循环直至用户输入数字大于5。要跳出这个循环,返回到shell提示符下,需要使用break命令。

#!/bin/bashwhile:do echo -n \"输入 1 到 5 之间的数字:\" read aNum case $aNum in1|2|3|4|5) echo \"你输入的数字为 $aNum!\";;*) echo \"你输入的数字不是 1 到 5 之间的! 游戏结束\"break;;esacdone

执行以上代码,输出结果为:

输入1到5之间的数字:3你输入的数字为3!输入1到5之间的数字:7你输入的数字不是1到5之间的!游戏结束

continue

continue命令与break命令类似,只有一点差别,它不会跳出所有循环,仅仅跳出当前循环。 对上面的例子进行修改:

#!/bin/bashwhile:do echo -n \"输入 1 到 5 之间的数字: \" read aNum case $aNum in1|2|3|4|5) echo \"你输入的数字为 $aNum!\";;*) echo \"你输入的数字不是 1 到 5 之间的!\"continue echo \"游戏结束\";;esacdone

运行代码发现,当输入大于5的数字时,该例中的循环不会结束,语句 echo \"游戏结束\" 永远不会被执行。

esac

case的语法和C family语言差别很大,它需要一个esac(就是case反过来)作为结束标记,每个case分支用右圆括号,用两个分号表示break。

Shell 函数

shell中函数的定义格式如下:

[function] funname [()]{ action;[returnint;]}

说明:

 

1、可以带function fun() 定义,也可以直接fun() 定义,不带任何参数。

2、参数返回,可以显示加:return 返回,如果不加,将以最后一条命令运行结果,作为返回值。 return后跟数值n(0-255)

函数返回值在调用该函数后通过 $? 来获得。

注意:所有函数在使用前必须定义。这意味着必须将函数放在脚本开始部分,直至shell解释器首次发现它时,才可以使用。调用函数仅使用其函数名即可。

函数参数

在Shell中,调用函数时可以向其传递参数。在函数体内部,通过 $n 的形式来获取参数的值,例如,$1表示第一个参数,$2表示第二个参数.

注意,$10 不能获取第十个参数,获取第十个参数需要${10}。当n>=10时,需要使用${n}来获取参数。

另外,还有几个特殊字符用来处理参数:

参数处理 $# $* $$ $! $@ $- $?

说明 传递到脚本的参数个数 以一个单字符串显示所有向脚本传递的参数 脚本运行的当前进程ID号 后台运行的最后一个进程的ID号 与$*相同,但是使用时加引号,并在引号中返回每个参数。 显示Shell使用的当前选项,与set命令功能相同。 显示最后命令的退出状态。0表示没有错误,其他任何值表明有错误。 Shell输入输出重定向

重定向命令列表如下:

命令 command > file command < file command >> file n > file n >> file n >& m n <& m << tag 说明 将输出重定向到 file。 将输入重定向到 file。 将输出以追加的方式重定向到 file。 将文件描述符为 n 的文件重定向到 file。 将文件描述符为 n 的文件以追加的方式重定向到 file。 将输出文件 m 和 n 合并。 将输入文件 m 和 n 合并。 将开始标记 tag 和结束标记 tag 之间的内容作为输入。 需要注意的是文件描述符 0 通常是标准输入(STDIN),1 是标准输出(STDOUT),2 是标准错误输出(STDERR)。

command1 < infile > outfile

同时替换输入和输出,执行command1,从文件infile读取内容,然后将输出写入到outfile中。

重定向深入讲解

一般情况下,每个 Unix/Linux 命令运行时都会打开三个文件:

  

标准输入文件(stdin):stdin的文件描述符为0,Unix程序默认从stdin读取数据。 标准输出文件(stdout):stdout 的文件描述符为1,Unix程序默认向stdout输出数据。 标准错误文件(stderr):stderr的文件描述符为2,Unix程序会向stderr流中写入错误信息。

默认情况下,command > file 将 stdout 重定向到 file,command < file 将stdin 重定向到 file。

2 表示标准错误文件(stderr)。

如果希望将 stdout 和 stderr 合并后重定向到 file,可以这样写:

$ command > file 2>&1 或者 $ command >> file 2>&1

如果希望对 stdin 和 stdout 都重定向,可以这样写:

$ command < file1 >file2

command 命令将 stdin 重定向到 file1,将 stdout 重定向到 file2。

Here Document

Here Document 是 Shell 中的一种特殊的重定向方式,用来将输入重定向到一个交互式 Shell 脚本或程序。 它的基本的形式如下:

command << delimiter document delimiter

它的作用是将两个 delimiter 之间的内容(document) 作为输入传递给 command。

注意:

结尾的delimiter 一定要顶格写,前面不能有任何字符,后面也不能有任何字符,包括空格和 tab 缩进。 

开始的delimiter前后的空格会被忽略掉。

/dev/null 文件

如果希望执行某个命令,但又不希望在屏幕上显示输出结果,那么可以将输出重定向到 /dev/null:

$ command > /dev/null

/dev/null 是一个特殊的文件,写入到它的内容都会被丢弃;如果尝试从该文件读取内容,那么什么也读不到。但是 /dev/null 文件非常有用,将命令的输出重定向到它,会起到\"禁止输出\"的效果。

如果希望屏蔽 stdout 和 stderr,可以这样写:

$ command > /dev/null 2>&1

注意:0 是标准输入(STDIN),1 是标准输出(STDOUT),2 是标准错误输出(STDERR)。

笔记

$ command > file 2>&1 $ command >> file 2>&1

这里的&没有固定的意思

放在>后面的&,表示重定向的目标不是一个文件,而是一个文件描述符,内置的文件描述符如下

1 => stdout 2 => stderr 0 => stdin

换言之 2>1 代表将stderr重定向到当前路径下文件名为1的regular file中,而2>&1代表将stderr重定向到文件描述符为1的文件(即/dev/stdout)中,这个文件就是stdout在file system中的映射

而&>file是一种特殊的用法,也可以写成>&file,二者的意思完全相同,都等价于 >file 2>&1

此处&>或者>&视作整体,分开没有单独的含义 顺序问题:

find /etc -name .bashrc > list 2>&1 # 我想问为什么不能调下顺序,比如这样 find /etc -name .bashrc 2>&1 > list

这个是从左到右有顺序的 第一种

xxx > list 2>&1

先将要输出到stdout的内容重定向到文件,此时文件list就是这个程序的stdout,再将stderr重定向到stdout,也就是文件list 第二种

xxx 2>&1 > list

先将要输出到stderr的内容重定向到stdout,此时会产生一个stdout的拷贝,作为程序的stderr,而程序原本要输出到stdout的内容,依然是对接在stdout原身上的,因此第二步重定向stdout,对stdout的拷贝不产生任何影响

因篇幅问题不能全部显示,请点此查看更多更全内容