zl程序教程

您现在的位置是:首页 >  其他

当前栏目

shell流程控制(流程不可为空、if else流程、for循环、while循环/无限循环、until循环、分支结构case...esac、跳出循环)、shell 函数(定义需在开头、如何调用、有无return返回值差别、函数参数$1 / ${10}、$? 获取返回值及其限制)、shell 输入输出重定向、Here Document重定向方式、/dev/null文件作用、shell文件包含及使用

2023-09-11 14:19:54 时间

一、shell流程控制

1、和其他语言不一样,sh 的流程控制不可为空。如果 else 分支没有语句执行,就不要写这个 else。

2、if else 流程

(1)if 语句语法格式:

if condition
then
    command1 
    command2
    ...
    commandN 
fi

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

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

末尾的 fi 就是 if 倒过来拼写,表示 if 结束。

(2)if else 语法格式:

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

(3)if else-if else 语法格式:

if condition1
then
    command1
elif condition2 
then 
    command2
else
    commandN
fi
#以下实例判断两个变量是否相等:
a=10
b=20
if [ $a == $b ]
then
   echo "a 等于 b"
elif [ $a -gt $b ]
then
   echo "a 大于 b"
elif [ $a -lt $b ]
then
   echo "a 小于 b"
else
   echo "没有符合的条件"
fi

#输出结果:a 小于 b

3、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

#顺序输出字符串中的字符:
#!/bin/bash
for str in This is a string
do
    echo $str
done

4、while 循环  ——  用于不断执行一系列命令,也用于从输入文件中读取数据。其语法格式为:

while condition
do
    command
done
#while循环可用于读取键盘信息。下面的例子中,输入信息被设置为变量FILM,按<Ctrl-D>结束循环。
echo '按下 <CTRL-D> 退出'
echo -n '输入你最喜欢的网站名: '
while read FILM
do
    echo "是的!$FILM 是一个好网站"
done

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

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

until condition
do
    command
done

5、分支结构 case ... esac

(1)case ... esac 为多选择语句,与其他语言中的 switch ... case 语句类似,是一种多分枝选择结构,每个 case 分支用右圆括号开始,用两个分号 ;; 表示 break,即执行结束,跳出整个 case ... esac 语句,esac(就是 case 反过来)作为结束标记。

(2)可以用 case 语句匹配一个值与一个模式,如果匹配成功,执行相匹配的命令。case ... esac 语法格式如下:

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

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

(3)取值将检测匹配的每一个模式。一旦模式匹配,则执行完匹配模式相应命令后不再继续其他模式。如果无一匹配模式,使用星号 * 捕获该值,再执行后面的命令。

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

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

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

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

二、shell 函数

1、linux shell 可以用户定义函数,然后在shell脚本中可以随便调用。shell中函数的定义格式如下:

[ function ] funname [()]
{
    action;
    [return int;]
}

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

# 1、定义了一个函数并进行调用,没有return,使用最后一行语句运行结果的返回值
demoFun(){
    echo "这是我的第一个 shell 函数!"
}
demoFun
# 这是我的第一个 shell 函数!

# 2、定义一个带有return语句的函数,使用 return 的结果作为返回值
funWithReturn(){
    echo "输入第一个数字: "
    read aNum
    echo "输入第二个数字: "
    read anotherNum
    echo "两个数字分别为 $aNum 和 $anotherNum !"
    return $(($aNum+$anotherNum))
}
funWithReturn
echo "输入的两个数字之和为 $? !"

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

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

2、函数参数:

(1)在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

#结果
第一个参数为 1 !
第二个参数为 2 !
第十个参数为 10 !
第十个参数为 34 !
第十一个参数为 73 !
参数总数有 11 个!
作为一个字符串输出所有参数 1 2 3 4 5 6 7 8 9 34 73 !

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

3、$? 仅对其上一条指令负责,一旦函数返回后其返回值没有立即保存入参数,那么其返回值将不再能通过 $? 获得。比如:

demoFun2
echo $?
demoFun1
echo 在这里插入命令!
echo $?

  调用 demoFun2 后,函数最后一条命令 expr 1 + 1 得到的返回值($?值)为 0,意思是这个命令没有出错(需要注意的是:没有使用 return,其返回值就不是 2 哦)。所有的命令的返回值仅表示其是否出错,而不会有其他有含义的结果

  第二次调用 demoFun1 后,没有立即查看 $? 的值,而是先插入了一条别的 echo 命令,最后再查看 $? 的值得到的是 0,也就是上一条 echo 命令的结果,而 demoFun1 的返回值被覆盖了。

  下面这个测试,连续使用两次 echo $?,得到的结果不同,更为直观:

#!/bin/bash
function demoFun1(){
    echo "这是我的第一个 shell 函数!"
    return `expr 1 + 1`
}

demoFun1
echo $?  # 显示的是 函数 demoFun1 的返回值 2
echo $?  # 显示的是 语句 echo $? 的返回值 0 ,代表运行没有报错

#输出结果:
这是我的第一个 shell 函数!
2
0

4、函数与命令的执行结果可以作为条件语句使用。要注意的是:shell 语言中 0 代表 true,0 以外的值代表 false

三、shell 输入/输出重定向

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

  重定向命令列表如下:

命令说明
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)。

2、输出重定向command1 > file1 (执行command1然后将输出的内容存入file1)

  注意:任何file1内的已经存在的内容将被新内容替代。如果要将新内容添加在文件末尾,请使用>>操作符。

3、输入重定向command1 < file1 (本来需要从键盘获取输入的命令会转移到文件读取内容)

4、重定向深入讲解:

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

  如果希望 stderr 重定向到 file,可以这样写:command 2>file  (stderr 追加到 file 文件末尾,可以这样写:command 2>>file)

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

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

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

command << delimiter
    document
delimiter

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

  注意:(1)结尾的delimiter 一定要顶格写,前面不能有任何字符,后面也不能有任何字符,包括空格和 tab 缩进。(2)开始的delimiter前后的空格会被忽略掉。

  这个我们在项目里制作 postgresql 镜像运行初始化脚本的时候有用到,当时不懂,现在了解了原来就是这个。

# 在命令行中通过 wc -l 命令计算 Here Document 的行数:
$ wc -l << EOF
    欢迎来到
    www.runoob.com
EOF
2          # 输出结果为 2 行

# 将 Here Document 用在脚本中
cat << EOF
欢迎来到
www.runoob.com
EOF

#输出结果:
欢迎来到
www.runoob.com

6、/dev/null 文件  ——   command > /dev/null

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

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

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

$ command > /dev/null 2>&1

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

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

四、文件包含

1、Shell 也可以包含外部脚本。这样可以很方便的封装一些公用的代码作为一个独立的文件。Shell 文件包含的语法格式如下:

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

2、实例展示

# test1.sh 代码如下:

#!/bin/bash
url="http://www.runoob.com"

3 test2.sh 代码如下:

#!/bin/bash
#使用 . 号来引用test1.sh 文件
. ./test1.sh

# 或者使用以下包含文件代码
# source ./test1.sh

echo "官网地址:$url"

  接下来,我们为 test2.sh 添加可执行权限并执行:

$ chmod +x test2.sh
$ ./test2.sh

#官网地址:http://www.runoob.com

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