shell脚本语言

Shell脚本语言

1.开始

vim test.sh
#!/bin/bash
echo "Hello World !"

#! 是一个约定的标记,它告诉系统这个脚本需要什么解释器来执行,即使用哪一种 Shell。

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

2.运行 Shell 脚本的两种方法:

一、
chmod + x ./test.sh  #使脚本具有执行权限
./test.sh  #执行脚本

注意,一定要写成 ./test.sh,而不是 test.sh,运行其它二进制的程序也一样,直接写 test.sh,linux 系统会去 PATH 里寻找有没有叫 test.sh 的,而只有 /bin, /sbin, /usr/bin,/usr/sbin 等在 PATH 里,你的当前目录通常不在 PATH 里,所以写成 test.sh 是会找不到命令的,要用 ./test.sh 告诉系统说,就在当前目录找。

二、
/bin/sh test.sh

这种运行方式是,直接运行解释器,其参数就是 shell 脚本的文件名
这种方式运行的脚本,不需要在第一行指定解释器信息,写了也没用。

3.Shell变量

3.1定义变量

####################定义变量####################
定义变量时,变量名不加美元符号, 如:
your_name="daitoue"
注意,变量名和等号之间不能有空格。同时,变量名的命名须遵循如下规则:
命名只能使用英文字母,数字和下划线,首个字符不能以数字开头。
中间不能有空格,可以使用下划线(_)。
不能使用标点符号。
不能使用bash里的关键字(可用help命令查看保留关键字)。

除了显式地直接赋值,还可以用语句给变量赋值,如:
for file in `ls /etc`
或
for file in $(ls /etc)

3.2使用变量

####################使用变量####################
使用一个定义过的变量,只要在变量名前面加美元符号即可,如:
for skill in Ada Coffe Action Java; do
    echo "I am good at ${skill}Script"
done

变量名外面的花括号是可选的,加不加都行,加花括号是为了帮助解释器识别变量的边界
建议使用  ${name}

3.3只读变量

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

下面的例子尝试更改只读变量,结果报错:
#!/bin/bash
myUrl="http://www.google.com"
readonly myUrl
myUrl="http://www.baidu.com"

3.4删除变量

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

3.5变量类型

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

1) 局部变量 局部变量在脚本或命令中定义,仅在当前shell实例中有效,其他shell启动的程序不能访问局部变量。
2) 环境变量 所有的程序,包括shell启动的程序,都能访问环境变量,有些程序需要环境变量来保证其正常运行。必要的时候shell脚本也可以定义环境变量。

环境变量可以在命令行中设置,但用户注销时这些值将丢失
环境变量均为大写
必须用export命令导出
设置环境变量:NAME=value
            export NAME(环境变量名大写)
显示环境变量:env可以查看所有环境变量
            echo $环境变量名
清除环境变量:unset 环境变量名


3) shell变量 shell变量是由shell程序设置的特殊变量。shell变量中有一部分是环境变量,有一部分是局部变量,这些变量保证了shell的正常运行

4.Shell中的特殊符号

1.注释符#
    除了#! /bin/bash里的#特殊
2.美元符$
    变量符。与反斜杠转义符相反,使其后的普通字符作为变量名,如$a表示变量a的值
3.单引号''
    被引起的字符全部做普通字符,即全部原样
    如:echo 'my $SHELL'
4.双引号""
    引号内的内容,除$、转义符\、倒引号``这三个保留特殊功能,其他字符均做普通字符
5.倒引号(反引号)``
    引号内的字符串当做shell命令行解释执行,得到的结果取代整个倒引号括起来的部分。
6.反斜线\
    反斜线是转义字符,他能把特殊字符编程普通字符,在某个字符前面利用反斜线能够阻止shell把后面的字符解释为特殊字符。

5.字符串操作

5.1拼接字符串

your_name="world"
# 使用双引号拼接
greeting="hello, "$your_name" !"
greeting_1="hello, ${your_name} !"
echo $greeting  $greeting_1

# 使用单引号拼接
greeting_2='hello, '$your_name' !' 
greeting_3='hello, ${your_name} !'
echo $greeting_2  $greeting_3

输出结果:
hello, world ! hello, world !
hello, world ! hello, ${your_name} !

5.2获取字符串长度

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

5.3提取子字符串

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

string="hahaha"
echo ${string:1:3} # 输出 aha

5.3查找子字符串

查找字符 h 或 a 的位置(哪个字母先出现就计算哪个):

string="xixihaha"
echo `expr index "$string" ha`  # 输出 5

6.Shell数组

bash支持一维数组(不支持多维数组),并且没有限定数组的大小。

类似于 Python 和 C 语言,数组元素的下标由 0 开始编号。获取数组中的元素要利用下标,下标可以是整数或算术表达式,其值应大于或等于 0。

6.1定义数组

在 Shell 中,用括号来表示数组,数组元素用"空格"符号分割开。定义数组的一般形式为:
        数组名=(值1 值2 ... 值n)
    如:
        array_name=(value0 value1 value2 value3)
    或:
        array_name=(
        value0
        value1
        value2
        value3
        )
    还可以单独定义数组的各个分量:
        array_name[0]=value0
        array_name[1]=value1
        array_name[n]=valuen
        
可以不使用连续的下标,而且下标的范围没有限制。

6.2读取数组

读取数组元素值的一般格式是:
    ${数组名[下标]}
    
使用 @ 符号可以获取数组中的所有元素,例如:
    echo ${array_name[@]} 或 echo ${array_name[*]}

6.3获取数组的长度

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

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

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

7.Shell注释

7.1单行注释

以"#"开头的行就是注释,会被解释器忽略。

sh里没有多行注释,只能每一行加一个#号。

7.2多行注释

如果在开发过程中,遇到大段的代码需要临时注释起来,过一会儿又取消注释,怎么办呢?

每一行加个#符号太费力了,可以把这一段要注释的代码用一对花括号括起来,定义成一个函数,没有地方调用这个函数,这块代码就不会执行,达到了和注释一样的效果。

shell本没有多行注释

多行注释还可以使用以下格式:

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

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

8.Shell传递参数

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

例:以下实例我们向脚本传递三个参数,并分别输出,其中 $0 为执行的文件名:

echo "Shell 传递参数实例!";
echo "执行的文件名:$0";
echo "第一个参数为:$1";
echo "第二个参数为:$2";
echo "第三个参数为:$3";

执行: ./test.sh 1 2 3

结果:   Shell 传递参数实例!
        执行的文件名:./test.sh
        第一个参数为:1
        第二个参数为:2
        第三个参数为:3
        


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

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




$* 与 $@ 区别:

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

9.Shell基本运算符

Shell 和其他编程语言一样,支持多种运算符,包括:
    算数运算符
    关系运算符
    布尔运算符
    字符串运算符
    文件测试运算符
    
原生bash不支持简单的数学运算,但是可以通过其他命令来实现,例如 awk 和 expr,expr 最常用。

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

例如,两个数相加(注意使用的是反引号 ` 而不是单引号 '):
#!/bin/bash
val=`expr 2 + 2`
echo "两数之和为 : $val"
结果:   两数之和为 : 4

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

9.1算数运算符

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

运算符                     说明                          举例
+                   加法  `expr $a + $b`              结果为 30。
-                   减法  `expr $a - $b`              结果为 -10。
*                   乘法  `expr $a \* $b`             结果为  200。
/                   除法  `expr $b / $a`              结果为 2。
%                   取余  `expr $b % $a`              结果为 0。
=                   赋值  a=$b 将把变量 b 的值赋给 a。
==          相等。用于比较两个数字,相同则返回 true。 [ $a == $b ] 返回 false。
!=          不相等。用于比较两个数字,不相同则返回 true。   [ $a != $b ] 返回 true。
注意:条件表达式要放在方括号之间,并且要有空格,例如: [$a==$b] 是错误的,必须写成 [ $a == $b ]
    乘号(*)前边必须加反斜杠(\)才能实现乘法运算;
    在 MAC 中 shell 的 expr 语法是:$((表达式)),此处表达式中的 "*" 不需要转义符号 "\" 

如:
if [ $a == $b ]
then
   echo "a 等于 b"
fi
if [ $a != $b ]
then
   echo "a 不等于 b"
fi
结果:a 不等于 b

除了最后两个其他不用放在[]中



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

9.2关系运算符

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

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

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

9.3布尔运算符

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

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

9.3逻辑运算符

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

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

9.4字符串运算符

下表列出了常用的字符串运算符,假定变量 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。

9.5文件测试运算符

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

10.Shell echo命令

Shell 的 echo 指令与 PHP 的 echo 指令类似,都是用于字符串的输出。命令格式:

echo string

10.1显示普通字符串

echo "It is a test"
这里的双引号完全可以省略,以下命令与上面实例效果一致:

echo It is a test

10.2显示转义字符

echo "\"It is a test\""

结果:
    "It is a test"

10.3显示变量

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

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

结果:
ubuntu@VM-0-4-ubuntu:~/Desktop$ ./test.sh
haha  (自己输入的)
haha is a test

10.4显示换行

echo -e "OK! \n" # -e 开启转义

10.5显示不换行

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

10.6显示结果定向至文件

echo "It is a test" > myfile

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

echo '$name\"'

结果:
    $name\"

10.8显示命令执行结果

echo `date`

注意: 这里使用的是反引号 `, 而不是单引号 '。

结果将显示当前日期
    Fri Aug 24 20:46:04 CST 2018

11.Shell printf 命令

printf 由 POSIX 标准所定义,因此使用 printf 的脚本比使用 echo 移植性好。

printf 使用引用文本或空格分隔的参数,外面可以在 printf 中使用格式化字符串,还可以制定字符串的宽度、左右对齐方式等。默认 printf 不会像 echo 自动添加换行符,我们可以手动添加 \n。

printf 命令的语法:

printf  format-string  [arguments...]
参数说明:
    format-string: 为格式控制字符串
    arguments: 为参数列表。
    
例:
    echo "Hello, Shell"   
    等同于
    printf "Hello, Shell\n"

11.1实例1

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位小数。

11.2实例2

# format-string为双引号
printf "%d %s\n" 1 "abc"

# 单引号与双引号效果一样 
printf '%d %s\n' 1 "abc" 

# 没有引号也可以输出
printf %s abcdef

# 格式只指定了一个参数,但多出的参数仍然会按照该格式输出,format-string 被重用
printf %s abc def

printf "%s\n" abc def

printf "%s %s %s\n" a b c d e f g h i j

# 如果没有 arguments,那么 %s 用NULL代替,%d 用 0 代替
printf "%s and %d \n" 

11.3printf的转义序列

序列  说明
\a  警告字符,通常为ASCII的BEL字符
\b  后退
\c  抑制(不显示)输出结果中任何结尾的换行字符(只在%b格式指示符控制下的参数字符串中有效),而且,任何留在参数里的字符、任何接下来的参数以及任何留在格式字符串中的字符,都被忽略
\f  换页(formfeed)
\n  换行
\r  回车(Carriage return)
\t  水平制表符
\v  垂直制表符
\\  一个字面上的反斜杠字符
\ddd    表示1到3位数八进制值的字符。仅在格式字符串中有效
\0ddd   表示1到3位的八进制值字符


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

d: Decimal 十进制整数 -- 对应位置参数必须是十进制整数,否则报错!

s: String 字符串 -- 对应位置参数必须是字符串或者字符型,否则报错!

c: Char 字符 -- 对应位置参数必须是字符串或者字符型,否则报错!

f: Float 浮点 -- 对应位置参数必须是数字型,否则报错!

如:其中最后一个参数是 "def",%c 自动截取字符串的第一个字符作为结果输出。

12Shell test命令

Shell中的 test 命令用于检查某个条件是否成立,它可以进行数值、字符和文件三个方面的测试。

12.1数值测试

参数          说明
-eq         等于则为真
-ne         不等于则为真
-gt         大于则为真
-ge         大于等于则为真
-lt         小于则为真
-le         小于等于则为真

例:
num1=100
num2=100
if test $[num1] -eq $[num2]
then
    echo '两个数相等!'
else
    echo '两个数不相等!'
fi
输出结果:
    两个数相等!

12.2字符串测试

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

例:
num1="haha"
num2="xixi"
if test $num1 = $num2
then
    echo '两个字符串相等!'
else
    echo '两个字符串不相等!'
fi

输出结果:
两个字符串不相等!

12.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
输出结果:

至少有一个文件存在!

13.Shell 流程控制

13.1 if else

if

if 语句语法格式:

if condition
then
    command1 
    command2
    ...
    commandN 
fi


写成一行(适用于终端命令提示符):

if [ $(ps -ef | grep -c "ssh") -gt 1 ]; then echo "true"; fi

if else

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

if elif else

语法格式:
if condition1
then
    command1
elif condition2 
then 
    command2
else
    commandN
fi


if else语句经常与test命令结合使用,如下所示:

num1=$[2*3]
num2=$[1+5]
if test $[num1] -eq $[num2]
then
    echo '两个数字相等!'
else
    echo '两个数字不相等!'
fi

13.2 for 循环

与其他编程语言类似,Shell支持for循环。

for循环一般格式为:

for var in item1 item2 ... itemN
do
    command1
    command2
    ...
    commandN
done

写成一行:

for var in item1 item2 ... itemN; do command1; command2… done;


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

例如,顺序输出当前列表中的数字:

for loop in 1 2 3 4 5
do
    echo "The value is: $loop"
done
输出结果:
The value is: 1
The value is: 2
The value is: 3
The value is: 4
The value is: 5

顺序输出字符串中的字符:

for str in 'This is a string'
do
    echo $str
done
输出结果:
    This is a string
    
for循环的其他方法:
for((assignment;condition:next));do
    command_1;
    command_2;
    commond_..;
done;

通常情况下 shell 变量调用需要加 $,但是 for 的 (()) 中不需要,下面来看一个例子:
#!/bin/bash
for((i=1;i<=5;i++));do
    echo "这是第 $i 次调用";
done;

注意:如果要在循环体中进行 for 中的 next 操作,记得变量要加 $,不然程序会变成死循环。

13.3while语句

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

while condition
do
    command
done


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

int=1
while(( $int<=5 ))
do
    echo $int
    let "int++"
done
结果:
1
2
3
4
5



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

while read FILM
do
    echo "是的!$FILM 是一个好网站"
done

13.4无限循环

无限循环语法格式:

while :
do
    command
done
或者

while true
do
    command
done
或者

for (( ; ; ))

13.4until循环

until 循环执行一系列命令直至条件为 true 时停止。
until 循环与 while 循环在处理方式上刚好相反。
一般 while 循环优于 until 循环,但在某些时候—也只是极少数情况下,until 循环更加有用。

until 语法格式:
until condition
do
    command
done

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

例:输出0~9的数字
a=0
until [ $a -ge 10 ]
do
   echo $a
   let "a++"
done

13.5 case

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

case语句格式如下:
case 值 in
模式1)
    command1
    command2
    ...
    commandN
    ;;
模式2)
    command1
    command2
    ...
    commandN
    ;;
esac

例:下面的脚本提示输入1到4,与每一种模式进行匹配:
echo '输入 1 到 4 之间的数字:'
echo '你输入的数字为:'
read aNum
case $aNum in
    1)  echo '你选择了 1'
    ;;
    2)  echo '你选择了 2'
    ;;
    3)  echo '你选择了 3'
    ;;
    4)  echo '你选择了 4'
    ;;
    *)  echo '你没有输入 1 到 4 之间的数字'
    ;;
esac
或(简便写法):
    1|2|3|4) echo "你选择了$aNum"
    ;;

13.6跳出循环

Shell使用两个命令来实现该功能:break和continue。

13.6.1 break命令

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

例:
while :
do
    read a
    case $a in
        1|2|3|4|5) echo "你选择了$a"
        ;;
        *) echo "no"
           break
        ;;
    esac
done

13.6.2 continue命令

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

13.7 esac

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

14.Shell函数

14.1函数定义

linux shell 可以用户定义函数,然后在shell脚本中可以随便调用。

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

[ function ] funname [()]

{

    action;

    [return int;]

}

说明:

1、可以带function fun() 定义,也可以直接fun() 定义,不带任何参数。
2、参数返回,可以显示加:return 返回,如果不加,将以最后一条命令运行结果,作为返回值。 return后跟数值n(0-255)

14.1.2 简单例子

例:定义了一个函数并调用

demoFun(){
    echo "这是我的第一个 shell 函数!"
}


echo "-----函数开始执行-----"
demoFun  #调用
echo "-----函数执行完毕-----"

14.1.3 带return的例子

funWithReturn(){
    echo "这个函数会对输入的两个数字进行相加运算..."
    echo "输入第一个数字: "
    read aNum
    echo "输入第二个数字: "
    read anotherNum
    echo "两个数字分别为 $aNum 和 $anotherNum !"
    return $(($aNum+$anotherNum))
}
funWithReturn
echo "输入的两个数字之和为 $? !"


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

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

14.2 函数参数

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

带参数的函数示例:
funWithParam(){
    echo "第一个参数为 $1 !"
    echo "第二个参数为 $2 !"
    echo "第十个参数为 $10 !"
    echo "第十个参数为 ${10} !"
    echo "第十一个参数为 ${11} !"
    echo "参数总数有 $# 个!"
    echo "作为一个字符串输出所有参数 $* !"
}
funWithParam 1 2 3 4 5 6 7 8 9 34 73

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

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

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

15.Shell 输入/输出重定向

15.1重定向命令

shell重定向命令列表如下:

命令                          说明
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 之间的内容作为输入。

需要注意的是文件描述符 0 通常是标准输入(STDIN),1 是标准输出(STDOUT),2 是标准错误输出(STDERR)。

15.2 输入重定向

和输出重定向一样,Unix 命令也可以从文件获取输入,语法为:
        command1 < file1
        
这样,本来需要从键盘获取输入的命令会转移到文件读取内容。

注意:输出重定向是大于号(>),输入重定向是小于号(<)。

我们需要统计 a.txt 文件的行数,命令行执行以下命令:
    wc -l a.txt   #结果:6 a.txt
也可以将输入重定向到 a.txt 文件:
    wc -l > a.txt #结果: 6
    
同时替换输入和输出,执行command1,从文件infile读取内容,然后将输出写入到outfile中:
    command1 < infile > outfile

15.3 重定向深入

一般情况下,每个 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。

#1.如果希望 stderr 重定向到 file,可以这样写:
        command 2 > file
    
    #2 表示标准错误文件(stderr)。

#2.如果希望将 stdout 和 stderr 合并后重定向到 file,可以这样写:
        command > file 2>&1

    或者

        command >> file 2>&1
    
#3.如果希望对 stdin 和 stdout 都重定向,可以这样写:
        command < file1 >file2
command 命令将 stdin 重定向到 file1,将 stdout 重定向到 file2

15.4 Here Document

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

它的基本的形式如下:

command << delimiter
    document
delimiter

它的作用是将两个 delimiter 之间的内容(document) 作为输入传递给 command。
注意:
结尾的delimiter 一定要顶格写,前面不能有任何字符,后面也不能有任何字符,包括空格和 tab 缩进。
开始的delimiter前后的空格会被忽略掉。

15.4.1 例子

#1.在命令行中通过 wc -l 命令计算 Here Document 的行数:
wc -l << EOF
aaaa
bbbb
cccc
EOF

#我们也可以将 Here Document 用在脚本中

15.5 /dev/null文件

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

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

#如果希望屏蔽 stdout 和 stderr,可以这样写:
    command > /dev/null 2>&1
#这里的&没有固定的意思,放在>后面的&,表示重定向的目标不是一个文件,而是一个文件描述符

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

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

16.Shell文件包含

#和其他语言一样,Shell 也可以包含外部脚本。这样可以很方便的封装一些公用的代码作为一个独立的文件。

Shell 文件包含的语法格式如下:

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

或

source filename
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,496评论 6 501
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,407评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,632评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,180评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,198评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,165评论 1 299
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,052评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,910评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,324评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,542评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,711评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,424评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,017评论 3 326
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,668评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,823评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,722评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,611评论 2 353

推荐阅读更多精彩内容

  • 官网 中文版本 好的网站 Content-type: text/htmlBASH Section: User ...
    不排版阅读 4,380评论 0 5
  • Shell就是一个命令行解释器,它的作用就是遵循一定的语法将输入的命令加以解释并传给系统。它为用户提供了一个向Li...
    Rathen阅读 18,788评论 0 10
  • Shell脚本语言学习一Shell脚本语言学习三 Shell脚本语言 echo命令? 显示普通字符串脚本代码: ...
    SimpleBK阅读 443评论 0 1
  • 自己写编译动态库脚本文件,脚本语言去操作数据库,插入数据,删除数据,修改数据等等... 内容一:Shell脚本语言...
    Jackey_song阅读 636评论 0 3
  • 从事机顶盒开发以来,看到很多的Shell脚本,比如:开机启动脚本、系统升级脚本、软探针脚本、和系统编译脚本。这些脚...
    五道杠小学生阅读 4,651评论 0 1