Shell 语法
Shell程序设计作为一种脚本语言,在Linux系统中有广泛的应用,本文记录了关于Shell程序设计的基础语法知识和常用命令,方便查询,熟练使用shell也需要经常实践,这对于完成一些较简单的编程任务很有帮助。
(1)变量
在shell里,使用变量之前并不需要事先做出声明,可以通过使用直接创建。默认情况下,所有的变量都被当做字符串进行存储,变量名区分大小写。变量名之前加一个$符号可以访问它的内容。
输入和输出:可以使用echo命令将一个变量的内容输出到终端,使用read命令(停下来等待用户输入)可以将用户的输入赋值给一个变量。
单引号和双引号:shell脚本的文件参数以空白符进行分割,如果一个参数包含空白符,就需要放到引号中(防止被解读为不同的参数)。双引号通常可以用来做字符串定界符。而像$salutation这样的变量放到引号中,有这几种情况:不加引号引用变量的值,放到双引号中也引用变量的值,放到单引号中就不引用,而是看作一个字符串。
$ salutation=Hello
$ echo $salutation # 输出 Hello
$ salutation="Yes Sir"
$ echo $salutation # 输出 Yes Sir
$ salutation=7+2
$ echo $salutation # 输出 7+2 (作为字符串看待)
# 使用引号:
$ myvar="Hello World"
$ echo $myvar # 输出 Hello World
$ echo "$myvar" # 输出 Hello World
$ echo '$myvar' # 输出 $myvar
$ echo $myvar # 转义字符,输出 $myvar
除了自定义变量外,shell还有一些环境变量和参数变量。
所谓环境变量,就是指脚本开始执行时,一些变量会根据当前环境设置中的值进行初始化。使用export命令可以设置环境变量。下面的示例展示了常用的环境变量:
$ echo $HOME # 当前用户的home目录,在我的机器上输出 /home/gzshan
$ echo $PATH # 以冒号分割的用来搜索命令的目录列表
$ echo $0 # shell脚本的名字
$ echo $# # 传递给脚本的参数数量,默认是0
$ echo $$ # shell脚本的进程号
所谓参数变量,就是指执行脚本程序如果带有参数,一些额外的变量会被创建,如下所示:
如:运行一个脚本的命令:./first.sh foo bar baz
$1 $2 $3 #分别指的就是参数foo,bar和baz
$# # 此时为3
$@ # 用于列出所有的参数变量,($* 也有相同功能,但受空白符影响,一般用$@)
(2)条件
条件判断是程序设计语言控制结构的基础,程序需要对条件进行测试和判断,从而执行不同的命令,完成不同的功能和任务。在shell脚本中,完成条件测试有两个命令:test 和 布尔判断命令 [ ]
-
test 命令
以下的例子展示了test命令的用法,注意如果then和if放在同一行则需要一个分号。
if test -f fred.c # 如果then写在这一行需要一个分号,这条语句检查一个文件是否存在 then echo "Hello" fi
-
布尔判断命令[ ]
把 [ 当作一条命令看起来有些奇怪,但是会使得程序变得简单。注意:使用 [ 时,后面必须有空格,还应该使用 ] 来结尾。
if [ -f fred.c ] ; then # 同样必须有分号,必须有空格 echo "test" fi
当使用以上两个命令时,可以使用的条件类型归结为三类,用下表归纳。
第一类:字符串的比较 | 比较结果 |
---|---|
string1 = string2 | 两个字符串相同结果为真 |
string1 != string2 | 两个字符串不相同结果为真 |
-n string | 字符串不为空则结果为真 |
-z string | 字符串为null则结果为真 |
第二类:算术比较 | 比较结果 |
expression1 -eq expression2 | 两个表达式结果相等为真 |
expression1 -nq expression2 | 两个表达式结果不相等为真 |
expression1 -gt expression2 | expression1 大于 expression2 为真 |
expression1 -ge expression2 | expression1 大于等于 expression2 为真 |
expression1 -lt expression2 | expression1 小于 expression2 为真 |
expression1 -le expression2 | expression1 小于等于 expression2 为真 |
!expression | expression为假则结果为真 |
第三类:与文件有关的条件测试 | 比较结果 |
-d file | 文件是一个目录,则结果为真 |
-f file | 文件是一个普通文件,则结果为真 |
-g file | 文件set-group-id 被设置,则结果为真 |
-u file | 文件set-user-id 被设置,则结果为真 |
-s file | 文件大小不为0,则结果为真 |
-r / -w / -x file | 文件可读 / 可写 / 可执行,则结果为真 |
-a file | 文件存在,则结果为为真 |
-c file | 文件存在并且为字符特殊文件,则结果为真 |
(3)控制结构
-
if 语句和elif 语句
if 语句比较简单,下面的例子展示了if 语句和elif语句的用法。
#!/bin/sh echo "Is it morning? Please answer yes or no" read timeofday if [ $timeofday = "yes" ] # if语句,后面跟条件测试 then echo "Good morning" elif [ $timeofday = "no" ]; then # then放同一行,加分号 echo "Good afternoon" else echo "Sorry,$timeofday not recognized. Enter yes or no" exit 1 fi exit 0
-
for 语句
for语句用以循环处理一组值,这组值可以是任意字符串的集合,也可以是其他命令的输出结果,下面用两个例子展示其用法。
#!/bin/sh for foo in bar fud 43 # 循环处理一组字符串 do echo $foo done exit 0
#!/bin/sh for file in $(ls f*.sh); do # 使用通配符扩展for循环,列出所有以f开头,扩展名为.sh的脚本文件 # 说明:$()是执行该命令得到的输出结果 lpr $file # lpr是打印命令 done exit 0
-
while 语句
如果事先不知道循环次数,for循环不太好使用的情况下,可以使用while循环。whie语句的do和done之间的语句反复执行,直到条件不再真为止。
#!/bin/sh echo "Enter password" read trythis while [ $trythis != "secret" ]; do # 反复执行,直到条件不再真为止 echo "sorry,try again" read trythis done exit 0
-
until 语句
until语句与while循环类似,所不同的是,until反复执行循环直到条件为真。
#!/bin/sh until who | grep "$1" > /dev/null do sleep 60 done echo -e 'a' echo "$1 has just logged in" exit 0
-
case 语句
case语句相比其他结构较为复杂,用下面的例子来介绍他的用法,需要特别注意的是:case按顺序查找第一个匹配的模式,而不是最佳匹配。
#!/bin/sh echo "Is it morning? Please answer yes or no" read timeofday case "$timeofday" in yes | y | Yes | YES ) echo "good morning";; # 注意每个模式末尾是两个分号 n* | N* ) echo "good afternoon";; * ) echo "sorry,answer not recognized";; esac exit 0
-
命令列表
有时需要将多个命令连接成一个序列,shell提供了命令列表,也就是and列表和or列表,类似于其他程序设计语言,它们也采用的是短路求值。
if [ -f file1 ] && echo "hello" && [ -f file2 ] && echo "here";then echo "in if" fi if [ -f file ] || echo "hello" || echo "here" ; then echo "in if" fi
-
语句块
在某些只允许使用单个语句的地方,要想使用多条语句,可以放到花括号中构建一个语句块。
get_confirm && { # and列表中使用语句块 echo "hello" cat test.txt }
(4)函数
要在shell脚本中使用函数,只需要写出函数名,然后跟一对括号,再把函数中的语句放在一对花括号中,并且把函数定义放到函数调用之前。函数体内可以用local关键字声明局部变量。
函数参数:当一个函数被调用时,脚本程序的位置参数($*,$@,$#,$1,$2
等等)会被替换为函数的参数,当函数执行完毕后,这些参数会恢复为先前的值。下面的例子展示了函数的用法。
#!/bin/sh
yes_or_no(){ # 函数定义
echo "Is your name $* ?"
while true
do
echo -n "Enter yes or no: "
read x
case "$x" in
y | yes ) return 0;;
n | no ) return 1;;
* ) echo "Answer yes or no"
esac
done
}
# 以下是主程序部分:
echo "Original parameters are $*"
if yes_or_no "$1"
then
echo "Hi $1,nice name"
else
echo "Never mind"
fi
exit 0
(5)命令
-
break命令和continue命令
这两条命令比较简单,break应用于跳出for、while、until循环,continue命令用于跳过当前这一次循环。
-
: 命令
:相当于一个空命令,或者相当于true的别名,例如 while : 就代表一个死循环
-
. 命令
. 命令用于在当前shell中执行命令,如前面用到的
./first.sh
-
echo 命令和read命令
echo命令用于输出结尾带有换行符的字符串,前面已多次用到,它还有两个常用的参数,如下所示。现在最新版本的shell常常用printf来代替echo。read命令用于将用户的输入赋给一个变量。
$ echo -n "string to output" # 去掉换行符 $ echo -e "string to outputc" # -e确保启用了反斜杠转移字符
-
eval 命令
eval命令用于对参数进行求值,它是shell的内置命令,通常不会以单独命令的形式存在,以下的例子展示eval的用法。eval命令就像一个额外的$,它给出一个变量的值的值。
foo=10 x=foo y='$'$x # $x就是foo echo $y # 输出的是$foo eval z='$'$x echo $z # 输出的是10
-
exec 命令
exec命令时执行一个shell程序,也就是将当前shell替换为一个不同的程序,当前脚本程序exec命令之后的代码都不会执行。还有一种用法是,它可以用于修改文件描述符,比较少见。
-
exit n 命令
exit命令使脚本程序以退出码n结束运行。退出码0表示成功,1~125是错误代码,126代表文件不可执行,127代表命令未找到。
-
export 命令
export命令将自己的参数建为一个环境变量,而这个环境变量可以被当前程序调用的其他脚本或者程序看到。
# 以下是export2.sh #!/bin/sh echo $foo echo $bar # 以下是export1.sh #!/bin/sh foo = "foo foo foo" export bar = "bar bar bar" export2 # 调用脚本2 # 此时如果执行脚本1,我们会看到输出bar的值,因为被声明为环境变量,在脚本2中可见,而foo不会被输出
-
expr 命令和
$( )
、$(( ))
expr命令将它的参数作为表达式来求值,最常见用法就是进行数学运算。$(command)的结果就是执行command的 输出结果,和两个反引号的功能相同。
对于表达式求值,一种更新的方法是使用
$(( ))
命令,将求值的表达式放到$(( ))
中,可以很简单的完成数学运算,常见的表达式求值有:加+、减-、乘*、除/、取模%、与&、或|、等于=、不等!=、大于>、小于<等等。以下的例子展示这几个命令的用法:
$ x=1 $ x=`expr $x + 1` # 反引号是x取值为xpr $x + 1的结果 $ x=$(expr $x + 1) # $(command)具有和反引号相同的功能 $ x=$(($x + 1)) # 一种更新的方法是使用$(( ))命令,代替expr命令
-
set 命令和unset 命令
set命令的作用是为shell设置参数变量。unset命令的作用是从环境中删除变量或者函数。
#!/bin/sh echo the date is $(date) set $(date) # 将date设置为参数变量 echo the month is $2 # 输出第二个位置参数,月份 foo = 0 unset foo # 删除变量foo exit 0
-
shift 命令
shift命令把所有的参数变量左移一个位置,使得
$2
变为$1
,$3
变为$2
,依次类推,原来的$1
被丢弃,而$0
仍将保持不变。如果指定数值参数,可以左移相应的次数。该命令的一个主要作用是用来扫描参数,如下所示。#!/bin/sh while [ "$1" != "" ]; do echo "$1" shift done exit 0
-
trap 命令
trap命令用于指定在接收到相应的信号后将要采取的行动,trap命令的常见形式如下:
trap command signal # 第一个参数是接收到信号时将要采取的行动 # 第二个参数是要处理的信号名
-
find 命令
find命令是一个很有用的命令,主要用于在系统中找文件,也就是文件搜索。其基本语法格式如下:
find [path] [options] [tests] [actions]
其中,path部分很好理解,是要搜索的路径,可以是相对路径,也可以是绝对路径,也可以是多个路径。
options部分是一些可用选项,主要有-depth(在查看目录本身之前先搜索目录的内容),-follow(跟随符号链接),-maxdepths N (最多搜索N层目录),-mount(或者-xdev,指不搜索其他文件系统中的目录)
tests是测试部分,每种测试的返回结果是true或false,主要有以下几种:-atime N (文件在N天前被最后访问过),-mtime N (文件在N天前被最后修改过),-name(文件名匹配),-newer otherfile(文件比otherfile要新),-type c(文件类型是c、d、f,分别对应特殊字符文件、目录、普通文件),-user username(文件的拥有者是username)。
注意:tests测试可以组合使用,有三个组合命令:-and,-or,-not
actions部分是匹配之后要执行的动作,比如:-exec command(执行一条命令,最常见),-print(打印)等等。
$ find / -name test -print $ find / -mount -name test -print $ find . -newer while2 -print $ find . -newer while2 -type f -print
-
grep 命令
grep命令是通用正则表达式解析器,通俗的说,find命令在系统中找文件,grep命令在文件中找字符串,一种常用的做法是将grep作为传递给-exec的一条命令。
grep命令的基本语法如下:
grep [options] PATTERN [FILES]
options是一些主要选项,常用的有-c (输出匹配行的数目)、-E(启用扩展表达式)、-h(取消每个输出行的普通前缀)、-i(忽略大小写)、-l(只列出包含匹配行的文件名)、-v(取反,搜索不匹配行)。
PATTERN主要是一些匹配模式,常用正则表达式来表示,关于正则表达式的内容参照另一篇博文正则表达式。
grep in words.txt grep -c in words.txt words2.txt grep -c -v in words.txt words2.txt grep "e$" words2.txt grep "a[[:blank:]]" words2.txt grep -E [a-z]{10} words2.txt
总结
Shell程序设计作为一种脚本语言,在Linux系统中有广泛的应用,本文记录了关于Shell程序设计的基础语法知识和常用命令,方便查询,熟练使用shell也需要经常实践,这对于完成一些较简单的编程任务很有帮助。另外,shell程序在Linux中海油一个可视化工具:dialog,由于不常使用,这里不再介绍。