Linux

Shell语法入门

开头

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)
  • ……

大家通常所说的Shell其实是 Bourne Again Shell,它也是大多数Linux 系统默认的 Shell。在一般情况下,人们并不区分 Bourne Shell 和 Bourne Again Shell,所以,像 #!/bin/sh,它同样也可以改为 #!/bin/bash。

#! 告诉系统其后路径所指定的程序即是解释此脚本文件的 Shell 程序。所以Shell脚本第一行通常是:

#!/bin/bash

注释

以 # 开头的行就是注释,会被解释器忽略。通过每一行加一个 # 号设置多行注释,像这样:

#--------------------------------------------
# 第一行
# 第二行
# 第三行
# 第四行
#--------------------------------------------

多行注释

:<<EOF
注释内容...
注释内容...
注释内容...
EOF

EOF 也可以使用其他符号:

:<<'
注释内容...
注释内容...
注释内容...
'

:<<!
注释内容...
注释内容...
注释内容...
!

变量

普通变量

变量名命名规则:

  • 命名只能使用英文字母,数字和下划线,首个字符不能以数字开头。
  • 中间不能有空格,可以使用下划线(_)。
  • 不能使用标点符号。
  • 不能使用bash里的关键字(可用help命令查看保留关键字)。

变量赋值:

变量名=值     # 注意 赋值语句两边不能有空格

赋值语句(即 “=” 号)两边不能有空格。等号右边若有空格的话,需要加上引号(单引号或双引号都是可以的)。

除了显式地直接赋值,还可以用语句给变量赋值,如:

for file in `ls /etc`
或
for file in $(ls /etc)

以上语句将 /etc 下目录的文件名循环出来。

只读变量

readonly variable_name

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

删除变量

unset variable_name

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

使用变量

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

pre_name="Helios"
echo "I am ${pre_name}_Fz"

打印结果为:"I am Helios_Fz"。在这里,如果不加 {} ,打印结果将会是 "I am $pre_name_Fz"。推荐给所有变量加上花括号,这是个好的编程习惯。

单引号与双引号:

单引号里的任何字符都会原样输出,单引号字符串中的变量是无效的;单引号字串中不能出现单独一个的单引号(对单引号使用转义符后也不行),但可成对出现,作为字符串拼接使用。

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

获取字符串长度:

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

提取子字符串:以下实例从字符串第 2 个字符开始截取 4 个字符:

string="runoob is a great site"
echo ${string:1:4} # 输出 unoo

注意:第一个字符的索引值为 0。

查找子字符串:查找字符 i 或 o 的位置(哪个字母先出现就计算哪个):

string="runoob is a great site"
echo `expr index "$string" io` # 输出 4

Shell传递参数

创建 test.sh 文件,内容如下:

m=$1
n=$2
echo $m-$n

执行命令:“sh test.sh aaa bbb”;输出 aaa-bbb

位置参数变量:

参数说明
$1、$2... 脚本程序的参数,分别代表程序的第1个参数、第2个参数、... 程序第10个以上的参数需要用大括号包含,如 ${10}
$*  代表命令行中的所有参数。在一个变量中将所有参数列出,各参数之间用环境变量 IFS 中的第一个字符分隔开。
$@ 和 $* 一样,也包含了命令行中的所有参数,但是不使用 IFS 环境变量,即使 IFS 为空,参数也是分开显示的。
如"$@"用 "" 括起来的情况、以"$1" "$2" … "$n" 的形式输出所有参数。
$! 后台运行的最后一个进程的ID号
$$ 脚本运行的当前进程ID号
$? 显示最后命令的退出状态。0表示没有错误,其他任何值表明有错误。

$* 与 $@ 区别:

  • 相同点:都是引用所有参数。
  • 不同点:只有在双引号中体现出来。假设在脚本运行时写了三个参数 1、2、3,,则 " * " 等价于 "1 2 3"(传递了一个参数),而 "@" 等价于 "1" "2" "3"(传递了三个参数)。
echo "-- $* 演示 ---"
for i in "$*"; do
    echo $i
done

echo "-- $@ 演示 ---"
for i in "$@"; do
    echo $i
done

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

$ chmod +x test.sh 
$ ./test.sh 1 2 3
-- $* 演示 ---
1 2 3
-- $@ 演示 ---
1
2
3

基本运算符

算数运算符

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

expr 是一款表达式计算工具,使用它能完成表达式的求值操作。

例如,两个数相加(注意使用的是反引号 ` 而不是单引号 '):

#!/bin/bash

val=`expr 2 + 2`
echo "两数之和为 : $val"

输出:“两数之和为 : 4”。

注意:

  • 表达式和运算符之间要有空格,例如 2+2 是不对的,必须写成 2 + 2,这与我们熟悉的大多数编程语言不一样。
  • 完整的表达式要被 ` ` 包含,注意这个字符不是常用的单引号,在 Esc 键下边。

关系运算符

运算符说明举例
+ 加法 `expr $a + $b`
- 减法 `expr $a - $b`
* 乘法 `expr $a * $b`
/ 除法 `expr $b / $a`
% 取余 `expr $b % $a`
= 赋值 a=$b 将把变量 b 的值赋给 a。
== 相等。用于比较两个数字,相同则返回 true。 [ $a == $b ] 
!= 不相等。用于比较两个数字,不相同则返回 true。 [ $a != $b ] 

注意:条件表达式要放在方括号之间,并且要有空格,例如: [$a==$b] 是错误的,必须写成 [ $a == $b ]。

布尔运算符

关系运算符只支持数字,不支持字符串,除非字符串的值是数字。

运算符说明举例
-eq 检测两个数是否相等,相等返回 true。 [ $a -eq $b ] 
-ne 检测两个数是否不相等,不相等返回 true。 [ $a -ne $b ] 
-gt 检测左边的数是否大于右边的,如果是,则返回 true。 [ $a -gt $b ] 
-lt 检测左边的数是否小于右边的,如果是,则返回 true。 [ $a -lt $b ] 
-ge 检测左边的数是否大于等于右边的,如果是,则返回 true。 [ $a -ge $b ] 
-le 检测左边的数是否小于等于右边的,如果是,则返回 true。 [ $a -le $b ] 

逻辑运算符

运算符说明举例
&& 逻辑的 AND

[[ $a -lt 100 && $b -gt 100 ]] 返回 false。

只有左边的“命令”执行成功,右边的“命令”才会被执行。

|| 逻辑的 OR

[[ $a -lt 100 || $b -gt 100 ]] 返回 true。

如果左边的“命令”未执行成功,就执行右边的“命令”。

字符串运算符

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

文件测试运算符

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

流程控制语句

if-else

if condition1
then
    statements1
elif condition2
then
    statements2
else
    statements3
fi

for

 for 循环将会遍历整个对象列表,依次执行每一个独立对象的循环内容。对象可以是命令行参数、文件名或是任何可以以列表形式建立的东西。其语法如下: 

for variable in values
do
    statements
done

case

每个 case 分支用右圆括号开始,用两个分号 ;; 表示 break,即执行结束,跳出整个 case ... esac 语句,esac(就是 case 反过来)作为结束标记。

case variable in
    pattern [ | pattern ] ... ) statements;;
    pattern [ | pattern ] ... ) statements;;
    ...
esac

while

while condition
do
    command
done

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

echo '按下 <CTRL-D> 退出'
echo -n '输入: '
while read FILM
do
    echo "输出: $FILM "
done

输出:

按下 <CTRL-D> 退出
输入: 1
输出: 1

无限循环:

while :
do
    command
done

或

while true
do
    command
done

until

until 循环执行一系列命令直至条件为 true 时停止。

until 循环与 while 循环在处理方式上刚好相反。

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

until 语法格式:

until condition
do
    command
done

跳出循环

break命令

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

continue

continue命令与break命令类似,只有一点差别,它不会跳出所有循环,仅仅跳出当前循环。

函数

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

[ function ] funname [()]
{
    action;

    [return int;]
}

不带返回例子:

#!/bin/bash

demoFun(){
    echo "这是一个 shell 函数!"
}
echo "-----函数开始执行-----"
demoFun
echo "-----函数执行完毕-----"

结果:

-----函数开始执行-----
这是一个 shell 函数!
-----函数执行完毕-----

带返回例子:

#!/bin/bash

funWithReturn(){
    echo "输入第一个数字: "
    read aNum
    echo "输入第二个数字: "
    read bNum
    echo "两个数字分别为 $aNum 和 $bNum !"
    return $(($aNum+$bNum))
}
funWithReturn
echo "输入的两个数字之和为 $? !"

结果:

输入第一个数字: 
1
输入第二个数字: 
2
两个数字分别为 1 和 2 !
输入的两个数字之和为 3 !

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

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

输入/输出重定向

大多数 UNIX 系统命令从终端接受输入并将所产生的输出发送回​​到终端。一个命令通常从一个叫标准输入的地方读取输入,将其输出写入到标准输出,默认情况下,标准输入和标准输出都是终端。

文件描述符:

  • 0 通常是标准输入 STDIN
  • 1 是标准输出 STDOUT
  • 2 是标准错误输出 STDERR 

重定向命令列表:

命令说明
command > file 将输出重定向到 file。
command < file 将输入重定向到 file。
command >> file 将输出以追加的方式重定向到 file。
n > file 将文件描述符为 n 的文件重定向到 file。
n >> file 将文件描述符为 n 的文件以追加的方式重定向到 file。
n >& m 将输出文件 m 和 n 合并。
n <& m 将输入文件 m 和 n 合并。
<< tag 将开始标记 tag 和结束标记 tag 之间的内容作为输入。

 /dev/null

/dev/null 称做空设备,是一个特殊的设备文件,它丢弃一切写入其中的数据,读取它则会立即得到一个EOF。空设备通常被用于丢弃不需要的输出流,

command > /dev/null

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

$ command > /dev/null 2>&1

这里的 2 和 > 之间不可以有空格,2> 是一体的时候才表示错误输出。

文件引用

Shell 文件引用的语法格式如下:

. filename # 注意点号(.)和文件名中间有一空格
或
source filename

例子:

test1.sh

#!/bin/bash

str="测试输出"

test2.sh

#!/bin/bash

. ./test1.sh
# source ./test1.sh

echo "$str"

执行test2.sh输出:

测试输出

注意:被包含的文件 test1.sh 不需要可执行权限。

原文地址:https://www.cnblogs.com/helios-fz/p/14339859.html