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 '按下 运行脚本,输出类似下面: 按下 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的拷贝不产生任何影响 因篇幅问题不能全部显示,请点此查看更多更全内容