#!/bin/sh
:<<EOF
000.定义变量
注意,变量名和等号之间不能有空格,这可能和你熟悉的所有编程语言都不一样。同时,变量名的命名须遵循如下规则:
命名只能使用英文字母,数字和下划线,首个字符不能以数字开头。
中间不能有空格,可以使用下划线 _。
不能使用标点符号。
不能使用bash里的关键字(可用help命令查看保留关键字)
注意:变量可以被重新赋值,如果不想被重新赋值,可以用只读变量
EOF
cur=12
tmp=good
name="nameString"
usr='high'
he="thingis"
#001.获取变量值赋值给其他变量后,输出
tak=$cur
strs=$tmp
str2=$name
str3=$usr
str4=${he}
echo "$cur ----- $tak"
echo "$tmp ----- $strs"
echo "$name ----- $str2"
echo "$usr ----- $str3"
echo "$he ----- $str4"
#002.创建只读变量
readonly zll=sunday
echo $zll
#003.修改其他变量的值
cur=23
name="goodbye"
echo $cur
echo $name
#004.撤销变量,销毁变量
unset cur
echo $cur #005.变量已被销毁,不可用,废语句,但是shell并不报错
#006.字符串:单引号,双引号,没引号, \"为转移字符,只能双引号用
cuname=tonghyh
taname="jack"
hi='sala'
astr="hello, good \"$taname\"! \n"
ahi='goodbye, '$hi''
echo $cuname $taname $hi
echo -e $astr
echo $ahi
#007.字符串拼接:双引号拼接,单引号拼接
get_0="Today is Tues, $taname"
get_1="Today is Godo, ${taname}"
get_2="Today is Hihi, \"$taname\""
get_3='Toogo ts GGGG, '$hi''
get_4='Toogo ts KKKK, '${hi}''
get_5='Toogo ts JJJJ, $hi'
echo $get_0
echo $get_1
echo $get_2
echo $get_3
echo $get_4
echo $get_5
#008.获取字符串长度
na="stringlessthing"
echo "na字符串长度是----${#na}"
echo ${#na[0]}
#009.提取子字符串:提取na串的第二个到第8个字符
#注意:第一个字符的索引值为0
echo ${na:1:7}
#010.在na串中查找字符h或g,先找到那个就返回位置数
#`是反引号运算符,不是单引号'
echo `expr index "$na" hg`
#011.定义数组
array_num=(1 2 3 4 5 6 7 8 9)
strs=(
Mon
Tus
Wen
Thr
Fri
Sta
Sun
)
nums[0]=11
nums[1]=12
nums[1000]=13
nums[2]=14
#注意:可以不使用连续的下标,而且下标的范围没有限制,但是建议规范连续使用
#012.读取数组
echo -e "读取数组\n"
echo ${array_num[0]} #读取单个元素
echo ${array_num[@]} #读取所有元素
echo ${nums[2]}
echo ${nums[@]}
echo ${strs[0]}
echo ${strs[@]}
echo "================here============="
for ds in $array_num;do
echo $ds
done
echo "=================hereEnd========="
#013.获取数组长度及单个元素长度
echo ${#array_num[@]}
echo ${#strs[*]}
echo -e "${#nums[0]} \n"
#014:注释
#单行注释用这个
##########
#多行注释
#可以这样
###########
:<<EOF
多行注释
也可以
这样
EOF
:<<!
也可以
写一个函数
然后不用
也是一个注释
!
#015.传递参数
#$0 为执行的文件名(包含文件路径)
#我们可以在执行 Shell 脚本时,向脚本传递参数,脚本内获取参数的格式为:$n。
#n代表一个数字,1为执行脚本的第一个参数,2为执行脚本的第二个参数,以此类推……
echo "传递参数实例-------^--^--"
echo "执行的文件名为---$0"
echo "传递的第一个参数是---$1"
echo "传递的第二个参数是---$2"
echo -e "传递的第三个参数是---$3 \n"
#016.几个特殊字符用来处理参数
:<<EOF
$# 传递到脚本的参数个数
$* 以一个单字符串显示所有向脚本传递的参数。
如"$*"用「"」括起来的情况、以"$1 $2 … $n"的形式输出所有参数。
$$ 脚本运行的当前进程ID号
$! 后台运行的最后一个进程的ID号
$@ 与$*相同,但是使用时加引号,并在引号中返回每个参数。
如"$@"用「"」括起来的情况、以"$1" "$2" … "$n" 的形式输出所有参数。
$- 显示Shell使用的当前选项,与set命令功能相同。
$? 显示最后命令的退出状态。0表示没有错误,其他任何值表明有错误。
EOF
echo "传递到脚本的参数个数---$#"
echo "脚本运行的当前进程ID号---$$"
echo "最后命令的退出状态---$?"
:<<!
$* 与 $@ 区别:
相同点:都是引用所有参数。
不同点:只有在双引号中体现出来。假设在脚本运行时写了
三个参数 1、2、3,则 " * " 等价于 "1 2 3"(传递了一个参数),
而 "@" 等价于 "1" "2" "3"(传递了三个参数)
!
echo "---\$*--和---\$@的区别演示---"
echo "####\$*###演示###"
for tmp in "$*";do
echo "$tmp"
done
echo "####\$@###演示###"
for t in "$@";do
echo "$t"
done
#017.数组操作及扩展
#Bash 支持关联数组,可以使用任意的字符串、或者整数作为下标来访问数组元素。
#关联数组使用 declare -A 命令来声明
declare -A site
site["baidu"]="www.baidu.com"
site["google"]="www.google.com"
site["yahu"]="www.yahu.com"
declare -A bay=(["bab"]="www.bab.com" ["dad"]="www.dad.com" ["jack"]="www.jack.com")
echo "#####关联数组####演示########"
echo ${site["baidu"]}
echo ${bay["dad"]}
#018.获取关联数组键值
echo "关联数组site的键为:${!site[@]}"
echo "关联数组bay的键为:${!bay[*]}"
echo "site的元素个数是:${#site[@]}"
echo "bay的元素个数为:${#bay[*]}"
:<<EOF
#019.算数运算符
+ 加法 `E $a + $b` 结果为 30。
- 减法 `E $a - $b` 结果为 -10。
* 乘法 `E $a \* $b` 结果为 200。
/ 除法 `E $b / $a` 结果为 2。
% 取余 `E $b % $a` 结果为 0。
= 赋值 a=$b 把变量 b 的值赋给 a。
== 相等。用于比较两个数字,相同则返回 true。 [ $a == $b ] 返回 false。
!= 不相等。用于比较两个数字,不相同则返回 true。 [ $a != $b ] 返回 true。
注意:条件表达式要放在方括号之间,并且要有空格,例如: [$a==$b] 是错误的,必须写成 [ $a == $b ]。
注意:E代表的是expr,这里是注释,用expr会报出语法错误
注意:
乘号(*)前边必须加反斜杠(\)才能实现乘法运算;
if...then...fi 是条件语句,后续将会讲解。
#在 MAC 中 shell 的 expr 语法是:$双括号中嵌套的表达式,此处表达式中的 "*" 不需要转义符号 "\"
EOF
hk=10
wk=30 #赋值
echo "#######算数运算符######演示######"
ak=`expr $hk + $wk` #加
echo "ak-------$ak"
bk=`expr $hk - $wk` #减
echo "bk-------$bk"
ck=`expr $hk \* $wk` #乘
echo "ck-------$ck"
dk=`expr $hk / $wk` #除
echo "dk-------$dk"
ek=`expr $hk % $wk` #取余
echo "ek-------$ek"
if [ $hk == $wk ];then
echo "hk 等于 wk"
fi
if [ $hk != $wk ];then
echo "hk 不等于 wk"
fi
:<<EOF
#020.关系运算符
运算符 说明 举例
-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。
EOF
readonly aa=10
readonly bb=20
echo "===关系运算符==="
if [ $aa -eq $bb ];then
echo "aa等于bb"
fi
if [ $aa -ne $bb ];then
echo "aa不等于bb"
fi
if [ $aa -gt $bb ];then
echo "aa大于bb"
fi
if [ $aa -lt $bb ];then
echo "aa小于bb"
fi
if [ $aa -ge $bb ];then
echo "aa大于等于bb"
fi
[ $aa -le $bb ] && {
echo "aa小于等于bb"
}
:<<EOF
#021.布尔运算符
运算符 说明 举例
! 非运算,表达式为 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。
EOF
readonly ab=10
readonly ac=20
readonly ad=0
echo "=====布尔运算符======"
[ $ab ] && {
echo "-----here_0-------"
}
[ !$ad ] && {
echo "-----here_1-------"
}
[ $ab -lt $ac -o $ab -eq $ac ] && {
echo "------here_2------"
}
[ $ab -eq $ac -a $ab -gt $ac ] && {
echo "-----here_3-------"
}
:<<EOF
#022.逻辑运算符
&& 逻辑的 AND [[ $a -lt 100 && $b -gt 100 ]] 返回 false
|| 逻辑的 OR [[ $a -lt 100 || $b -gt 100 ]] 返回 true
EOF
readonly ae=10
readonly af=20
echo "#####逻辑运算符#####"
if [[ $ae -lt 50 && $af -ge 20 ]];then
echo "ae小于50且af大于等于20"
fi
[ $ae -eq 10 ] && {
echo "ae 等于10"
}
if [[ $ae -gt 30 || af -lt 50 ]];then
echo "ae大于30或者af小于50"
fi
:<<EOF
#023.字符串运算符
下表列出了常用的字符串运算符,假定变量 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。
$ 检测字符串是否不为空,不为空返回 true。 [ $a ] 返回 true。
EOF
readonly strs0="abcdef"
readonly strs1="cdefgh"
readonly strs2="abcdef"
readonly strs3=
echo "###字符串运算###"
if [ $strs0 = $strs2 ];then
echo "strs0 = strs2"
fi
[ $strs0 != $strs1 ] && {
echo "strs0 != strs1"
}
if [ -z $strs3 ];then
echo "-z字符串strs3长度为零"
fi
if [ -n $strs0 ];then
echo "-n字符串strs0长度不为零"
fi
if [ $strs0 ];then
echo "字符串strs0不为空 $"
fi
:<<!
#024.文件测试运算符
文件测试运算符用于检测 Unix 文件的各种属性。
属性检测描述如下:
操作符 说明 举例
-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。
其他检查符:
-S: 判断某文件是否 socket。
-L: 检测文件是否存在并且是一个符号链接。
!
CURRENT_PATH=$(pwd)
file0="${CURRENT_PATH}/ltop"
file2="${CURRENT_PATH}/top.sh"
echo "###检测文件运算符###"
if [ -b $file0 ];then
echo "$file0 是块设备"
else
echo "$file0 不是块设备"
fi
if [ -d $file0 ];then
echo "$file0 是目录"
else
echo "$file0 不是目录"
fi
if [ -f $file2 ];then
echo "$file2 是普通文件"
else
echo "$file2 不是普通文件,是目录或者设备文件"
fi
if [ -L $file0 ];then
echo "$file0 是一个符号链接"
else
echo "$file0 不是一个符号链接"
fi
if [ -x $file2 ];then
echo "$file2 具有可执行权限"
else
echo "$file2 不具有可执行权限"
fi
if [ -e $file2 ];then
echo "$file2 存在此文件"
else
echo "$file2 不存在此文件"
fi
#025.echo命令
readonly cmd="string is long"
echo "###echo命令###"
#输出字符
echo $cmd
echo cmd
echo "it is good thing"
echo it is good thing
#输出转移字符
echo "\"it is good thing\""
#显示变量
#read 命令从标准输入中读取一行,并把输入行的每个字段的值指定给 shell 变量
read bot
echo "$bot is here"
#显示换行
echo -e "OK! \n"
#显示不换行
echo -e "OKK! \c"
echo "string is heer"
#输出结果定向到文件
echo "he is a boy" >my.txt
#原样输出字符串,不进行转义或取变量(用单引号)
echo '$cmd\"'
echo '$cmd'
#显示命令执行结果
echo `date`
a00.sh脚本 如上
内容摘要:
#000.定义变量
#001.获取变量值赋值给其他变量后,输出
#002.创建只读变量
#003.修改其他变量的值
#004.撤销变量,销毁变量
#005.变量已被销毁,不可用,废语句,但是shell并不报错
#006.字符串:单引号,双引号,没引号, \"为转移字符,只能双引号用
#007.字符串拼接:双引号拼接,单引号拼接
#008.获取字符串长度
#009.提取子字符串:提取na串的第二个到第8个字符
#010.在na串中查找字符h或g,先找到那个就返回位置数
#011.定义数组
#012.读取数组
#013.获取数组长度及单个元素长度
#014:注释
#015.传递参数
#016.几个特殊字符用来处理参数
#017.数组操作及扩展
#018.获取关联数组键值
#019.算数运算符
#020.关系运算符
#021.布尔运算符
#022.逻辑运算符
#023.字符串运算符
#024.文件测试运算符
#025.echo命令
输出结果:
zhangleilei@compile_8:~/bin/test_C/shell$ ./a00.sh
12 ----- 12
good ----- good
nameString ----- nameString
high ----- high
thingis ----- thingis
sunday
23
goodbye
tonghyh jack sala
hello, good "jack"!
goodbye, sala
Today is Tues, jack
Today is Godo, jack
Today is Hihi, "jack"
Toogo ts GGGG, sala
Toogo ts KKKK, sala
Toogo ts JJJJ, $hi
na字符串长度是----15
15
tringle
6
读取数组
1
1 2 3 4 5 6 7 8 9
14
11 12 14 13
Mon
Mon Tus Wen Thr Fri Sta Sun
================here=============
1
=================hereEnd=========
9
7
2
传递参数实例-------^--^--
执行的文件名为---./a00.sh
传递的第一个参数是---
传递的第二个参数是---
传递的第三个参数是---
传递到脚本的参数个数---0
脚本运行的当前进程ID号---12730
最后命令的退出状态---0
---$*--和---$@的区别演示---
####$*###演示###
####$@###演示###
#####关联数组####演示########
www.baidu.com
www.dad.com
关联数组site的键为:yahu google baidu
关联数组bay的键为:bab dad jack
site的元素个数是:3
bay的元素个数为:3
./a00.sh: line 206: E: command not found
./a00.sh: line 206: E: command not found
./a00.sh: line 206: E: command not found
./a00.sh: line 206: E: command not found
./a00.sh: line 206: E: command not found
#######算数运算符######演示######
ak-------40
bk--------20
ck-------300
dk-------0
ek-------10
hk 不等于 wk
===关系运算符===
aa不等于bb
aa小于bb
aa小于等于bb
=====布尔运算符======
-----here_0-------
-----here_1-------
------here_2------
#####逻辑运算符#####
ae小于50且af大于等于20
ae 等于10
ae大于30或者af小于50
###字符串运算###
strs0 = strs2
strs0 != strs1
-z字符串strs3长度为零
-n字符串strs0长度不为零
字符串strs0不为空 $
###检测文件运算符###
/home/zhangleilei/bin/test_C/shell/ltop 不是块设备
/home/zhangleilei/bin/test_C/shell/ltop 不是目录
/home/zhangleilei/bin/test_C/shell/top.sh 是普通文件
/home/zhangleilei/bin/test_C/shell/ltop 是一个符号链接
/home/zhangleilei/bin/test_C/shell/top.sh 具有可执行权限
/home/zhangleilei/bin/test_C/shell/top.sh 存在此文件
###echo命令###
string is long
cmd
it is good thing
it is good thing
"it is good thing"
nanananna
nanananna is here
OK!
OKK! string is heer
$cmd\"
$cmd
Fri Sep 22 08:40:56 UTC 2023
a01.sh脚本 如下
内容摘要
026.printf命令
026.test数值测试命令
027.test字符串测试命令
028.test文件测试命令
029.if语句
030.for语句
031.while循环
032.until循环
033.case语句
#!/bin/sh
:<<EOF
026.printf命令
printf 命令模仿 C 程序库(library)里的 printf() 程序。
printf 由 POSIX 标准所定义,因此使用 printf 的脚本比使用 echo 移植性好。
printf 使用引用文本或空格分隔的参数,外面可以在 printf 中使用格式化字符串,还可以制定字符串的宽度、
左右对齐方式等。默认的 printf 不会像 echo 自动添加换行符,我们可以手动添加 \n。
printf 命令的语法:
printf format-string [arguments...]
参数说明:
format-string: 为格式控制字符串
arguments: 为参数列表。
printf 的转义序列
序列 说明
\a 警告字符,通常为ASCII的BEL字符
\b 后退
\c 抑制(不显示)输出结果中任何结尾的换行字符(只在%b格式指示符控制下的参数字符串中有效),
而且,任何留在参数里的字符、任何接下来的参数以及任何留在格式字符串中的字符,都被忽略
\f 换页(formfeed)
\n 换行
\r 回车(Carriage return)
\t 水平制表符
\v 垂直制表符
\\ 一个字面上的反斜杠字符
\ddd 表示1到3位数八进制值的字符。仅在格式字符串中有效
\0ddd 表示1到3位的八进制值字符
EOF
echo "###printf命令###"
echo "hello world"
printf "hello world\n"
printf "%-10s %-8s %-4s\n" 姓名 性别 体重kg
printf "%-10s %-8s %-4.2f\n" 张三 男 80.1223
printf "%-10s %-8s %-4.2f\n" 李四 男 56.8945
printf "%-10s %-8s %-4.2f\n" 王五 男 39.6776
:<<!
%s %c %d %f 都是格式替代符,%s 输出一个字符串,%d 整型输出,%c 输出一个字符,%f 输出实数,以小数形式输出。
%-10s 指一个宽度为 10 个字符(- 表示左对齐,没有则表示右对齐),任何字符都会被显示在 10 个字符宽的字符内,
如果不足则自动以空格填充,超过也会将内容全部显示出来。
%-4.2f 指格式化为小数,其中 .2 指保留2位小数
printf 命令的语法:
printf format-string [arguments...]
参数说明:
format-string: 为格式控制字符串
arguments: 为参数列表。
!
printf "%d %s\n" 10 "astring"
printf '%d %s\n' 10 "astring"
printf %s abcdefg
#格式只指定了一个参数,但多出的参数仍然会按照该格式输出,format-string 被重用
printf %s hhh good this
#如果没有 arguments,那么 %s 用NULL代替,%d 用 0 代替
printf "%s and %d \n"
:<<EOF
026.test数值测试命令
Shell中的 test 命令用于检查某个条件是否成立,它可以进行数值、字符和文件三个方面的测试。
数值测试
参数 说明
-eq 等于则为真
-ne 不等于则为真
-gt 大于则为真
-ge 大于等于则为真
-lt 小于则为真
-le 小于等于则为真
EOF
readonly num_0=100
readonly num_1=200
echo "###test数值测试命令###"
if test $[num_0] -lt $[num_1];then
echo "$num_0小于$num_1"
else
echo "$num_0大于$num_1"
fi
#注意:代码中的 [] 执行基本的算数运算,等号两边不能有空格,如下:
result=$[num_0*num_1]
echo "result的值为:$result"
:<<EOF
027.test字符串测试命令
参数 说明
= 等于则为真
!= 不相等则为真
-z 字符串 字符串的长度为零则为真
-n 字符串 字符串的长度不为零则为真
EOF
echo "###test字符串测试命令###"
readonly strs0="jackoff"
readonly strs1="jackon"
if test $strs0 = $strs1;then
echo "两个字符串相等"
else
echo "两个字符串不相等"
fi
if test -n $strs0;then
echo "字符串长度不为零"
else
echo "字符串长度为零"
fi
:<<EOF
028.test文件测试命令
参数 说明
-e 文件名 如果文件存在则为真
-r 文件名 如果文件存在且可读则为真
-w 文件名 如果文件存在且可写则为真
-x 文件名 如果文件存在且可执行则为真
-s 文件名 如果文件存在且至少有一个字符则为真
-d 文件名 如果文件存在且为目录则为真
-f 文件名 如果文件存在且为普通文件则为真
-c 文件名 如果文件存在且为字符型特殊文件则为真
-b 文件名 如果文件存在且为块特殊文件则为真
EOF
CURRENT_PATH=$(pwd)
file0="${CURRENT_PATH}/top.sh"
file1="${CURRENT_PATH}/ltop"
echo "###test文件测试命令###"
if test -e $file0;then
echo "文件存在"
else
echo "文件不存在"
fi
if test -w $file1;then
echo "文件具有写权限"
else
echo "文件没有写权限"
fi
################################################################
#029.if语句
#
#语法格式1:
#if [ ];then
# ...
#else
# ...
#fi
#
#语法格式2:
#if [ ];then
# ...
#elif [ ];then
# ...
#elif [ ];then
# ...
#else
# ...
#fi
#
#语法格式3:
#if [ ]
#then
# ...
#elif [ ]
#then
# ...
#else
# ...
#fi
#
#语法格式4:
#if [ ]
#then
# ...
#else
# ...
#fi
#
#语法格式5:
#if test -z $strs0;then
# ...
#else
# ...
#fi
#
#语法格式6:
#if (( a > b ));then
# ...
#elif (( a == b ));then
# ...
#else
# ...
#fi
#如果使用 ((...)) 作为判断语句,大于和小于可以直接使用 > 和 <
#
#if (( a > b )); then
# ...
#fi
################################################################
echo "###if语句测试###"
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
str0=root
str1=testings
str2=God
#比较串是否一样, USER表示当前用户
if [ $USER == "$str0" ];then
echo "welcome root......"
else
echo "Sorry, you aren't root!"
fi
#比较字符串大小:大于、小于号必须加转移符号,否则会被认为是重定向 \> \<
if [ "$str1" \> "$str2" ];then
echo "$str1 大于 $str2"
else
echo "$str1 小于 $str2"
fi
#判断字符串长度是否非零:非零则为真,零则为假
if [ -n "$str3" ];then
echo "$str3 is not EMPTY!!"
else
echo "$str3 is EMPTY"
fi
#判断字符串是否为零:零则为真,非零则为假
if [ -z "$str3" ];then
echo "str3 is Zero"
else
echo "str3 is not Zero"
fi
if (( $b > $a ));then
echo "b大于a"
else
echo "a大于b"
fi
####################################################
#030.for语句
#
#语法格式1:
#for doo in 1 2 3 4;do
# ...
#done
#
#语法格式2:
#for doo in 1 2 3 4
#do
# ...
#done
###################################################
echo "###for循环###"
#循环遍历输出数字
for cur in 1 2 3 4 0
do
echo "The value is $cur"
done
#循环遍历输出字符串
for driver in This is a dog
do
echo "This value is $driver"
done
#嵌套if语句的for循环
CURRENT_PATH=$(pwd)
one="${CURRENT_PATH}/zll.sh"
DRIVER=hostapd.wlan0
for doo in $DRIVER; do (
if [ -x $one ];then
echo "zll.sh is x file"
sh $one
else
echo "zll.sh isn't x file"
fi
);done
:<<EOF
031.while循环
while 循环用于不断执行一系列命令,也用于从输入文件中读取数据。其语法格式为:
while condition;do
command
done
EOF
echo "###while循环测试###"
int=0
while(( int < 5 ))
do
echo $int
let "int++"
done
i=0
name="thing..."
while((i < 5));do
echo $name
let "i++"
done
#以上实例使用了 Bash let 命令,它用于执行一个或多个表达式,变量计算中不需要加上 $ 来表示变量,具体可查阅:Bash let 命令
:<<!
#while循环可用于读取键盘信息
echo '按下<CTRL-D>键退出'
echo -n '输入你喜欢的网站名:'
while read baa;do
echo "是的!$baa是一个好网站"
done
!
:<<EOF
无限循环模式
while :
do
command
done
或
while true;do
command
done
或
for (( ; ; ))
EOF
#
#while true;do
# echo $name
#done
:<<EOF
032.until循环
until 循环执行一系列命令直至条件为 true 时停止。
until 循环与 while 循环在处理方式上刚好相反。
一般 while 循环优于 until 循环,但在某些时候—也只是极少数情况下,until 循环更加有用。
until 语法格式:
until condition;do
command
done
注:condition 一般为条件表达式,如果返回值为 false,则继续执行循环体内的语句,否则跳出循环
EOF
ar=0
echo "###until循环###"
until (($ar > 10));do
echo $ar
ar=`expr $ar + 1`
done
:<<EOF
033.case语句
case ... esac 为多选择语句,与其他语言中的 switch ... case 语句类似,是一种多分支选择结构,
每个 case 分支用右圆括号开始,用两个分号 ;; 表示 break,即执行结束,跳出整个 case ... esac 语句,
esac(就是 case 反过来)作为结束标记。
可以用 case 语句匹配一个值与一个模式,如果匹配成功,执行相匹配的命令。
语法格式:
case 值 in
模式1)
command1
command2
...
commandN
;;
模式2)
command1
command2
...
commandN
;;
esac
case 工作方式如上所示,取值后面必须为单词 in,每一模式必须以右括号结束。
取值可以为变量或常数,匹配发现取值符合某一模式后,其间所有命令开始执行直至 ;;。
取值将检测匹配的每一个模式。一旦模式匹配,则执行完匹配模式相应命令后不再继续其他模式。
如果无一匹配模式,使用星号 * 捕获该值,再执行后面的命令
EOF
echo "###case语句###"
read ay
case $ay in
1)echo "你写了1"
echo it is good
;;
2)echo "你写了2"
echo it is good
;;
3)echo "你写了3"
echo it is good
;;
4)echo "你写了4"
echo it is good
;;
*)echo "你写的数字不在1-4之内"
echo it is not good
;;
esac
输出结果:
###printf命令###
hello world
hello world
姓名 性别 体重kg
张三 男 80.12
李四 男 56.89
王五 男 39.68
10 astring
10 astring
abcdefghhhgoodthis and 0
###test数值测试命令###
100小于200
result的值为:20000
###test字符串测试命令###
两个字符串不相等
字符串长度不为零
###test文件测试命令###
文件存在
文件具有写权限
###if语句测试###
a 小于 b
Sorry, you aren't root!
testings 大于 God
is EMPTY
str3 is Zero
b大于a
###for循环###
The value is 1
The value is 2
The value is 3
The value is 4
The value is 0
This value is This
This value is is
This value is a
This value is dog
zll.sh is x file
welcome to zll.sh
###while循环测试###
0
1
2
3
4
thing...
thing...
thing...
thing...
thing...
###until循环###
0
1
2
3
4
5
6
7
8
9
10
###case语句###
9
你写的数字不在1-4之内
it is not good
#!/bin/sh
:<<EOF
在循环过程中,有时候需要在未达到循环结束条件时强制跳出循环,Shell 使用两个命令来实现该功能:break 和 continue。
034.break 命令
break 命令允许跳出所有循环(终止执行后面的所有循环)。
echo "###break语句###"
while true;do
echo "输入1-5之间的数字"
read num
case $num in
1|2|3|4|5)echo "你输入的数字是$num"
echo "请继续..."
;;
*)echo "你输入的不是1-5"
echo "goodbye"
break
;;
esac
done
EOF
#035.continue语句
#continue 命令与 break 命令类似,只有一点差别,它不会跳出所有循环,仅仅跳出当前循环
#运行代码发现,当输入大于5的数字时,该例中的循环不会结束,语句 echo "游戏结束" 永远不会被执行
:<<EOF
echo "###continue语句###"
while true;do
echo "输入1-5之间的数字"
read num
case $num in
1|2|3|4|5)echo "你输入的数字是$num"
echo "请继续..."
;;
*)echo "你输入的不是1-5"
echo "goodbye"
continue
echo "游戏结束"
;;
esac
done
EOF
:<<EOF
036.shell函数
linux shell 可以用户定义函数,然后在shell脚本中可以随便调用。
shell中函数的定义格式如下:
[ function ] funname [()]
{
action;
[return int;]
}
说明:
1、可以带function fun() 定义,也可以直接fun() 定义,不带任何参数。
2、参数返回,可以显示加:return 返回,如果不加,将以最后一条命令运行结果,作为返回值。
return后跟数值n(0-255)
$? 显示最后命令的退出状态。0表示没有错误,其他任何值表明有错误。
EOF
echo "###shell函数###"
demofun(){
echo "这是一个没什么用的函数吧"
echo "func"
}
echo "调用函数demofun"
demofun
echo "调用完毕"
echo $?
:<<EOF
37.return 语句
函数返回值在调用该函数后通过 $? 来获得。
注意:所有函数在使用前必须定义。这意味着必须将函数放在脚本开始部分,
直至shell解释器首次发现它时,才可以使用。调用函数仅使用其函数名即可
EOF
echo "###带返回值的函数使用###"
addnum(){
echo "求和运算并返回和值"
echo "输入第一个参数是:"
read Firstnum
echo "输入第二个参数是:"
read Secondnum
echo "输入的两个参数分别是:$Firstnum 和 $Secondnum"
echo "两数之和是: $(($Firstnum+$Secondnum))"
return $(($Firstnum+$Secondnum))
}
addnum
echo "函数的返回值是 $?"
:<<EOF
38.函数参数
在Shell中,调用函数时可以向其传递参数。在函数体内部,
通过 $n 的形式来获取参数的值,例如,$1表示第一个参数,$2表示第二个参数...
注意,$10 不能获取第十个参数,获取第十个参数需要${10}。当n>=10时,需要使用${n}来获取参数。
另外,还有几个特殊字符用来处理参数:
参数处理 说明
$# 传递到脚本或函数的参数个数
$* 以一个单字符串显示所有向脚本传递的参数
$$ 脚本运行的当前进程ID号
$! 后台运行的最后一个进程的ID号
$@ 与$*相同,但是使用时加引号,并在引号中返回每个参数。
$- 显示Shell使用的当前选项,与set命令功能相同。
$? 显示最后命令的退出状态。0表示没有错误,其他任何值表明有错误。
EOF
echo "###函数参数运用###"
printf_para(){
echo "输入第1个参数是:$1"
echo "输入第5个参数是:$5"
echo "输入第9个参数是:$9"
echo "输入第10个参数是:$10"
echo "输入第10个参数是:${10}"
echo "输入第11个参数是:${11}"
echo "输入第11个参数是:$11"
echo "输入参数总个数为:$#个"
echo "输入的参数是:$*"
echo "输入的参数是:$@"
}
printf_para 1 2 3 4 5 6 7 8 9 10 11 12 16 18 19 20
#39.$?仅对其上一条指令负责,一旦函数返回后其返回值没有立即保存入参数,那么其返回值将不再能通过 $? 获得
echo "###$? 作用效果域演示###"
function addnumber(){
echo "这个还是加法运算..."
num_0=9
num_1=10
sk=`expr $num_0 + $num_1`
return $sk
}
addnumber
echo $?
echo "加入一个语句..."
echo $?
:<<EOF
#40.函数与命令的执行结果运用
函数与命令的执行结果可以作为条件语句使用。要注意的是,和 C 语言不同,
shell 语言中 0 代表 true,0 以外的值代表 false
如下:
grep 是从给定字符串中寻找匹配内容的命令。首先看出如果找到了匹配的内容,
会打印匹配部分且得到的返回值 $? 为 0,如果找不到,则返回值 $? 为 1
接下来分别将这两次执行的 grep 命令当作条件语句交给 if 判断,
得出返回值 $? 为 0,即执行成功时,条件语句为 true,当返回值 $? 为 1,即执行失败时,条件语句为 false。
之后再用函数的 return 值作为测试,其中 demoFun1 返回值为 0,
demoFun2 返回值选择了任意一个和 0 不同的整数,这里为 12。
将函数作为条件语句交给 if 判断,得出返回值为 0 时,依然为 true,
而返回值只要不是 0,条件语句都判断为 false
EOF
echo "###函数与命令的执行结果作为条件语句使用###"
echo "hello world" | grep -e hello
echo $?
echo "hello world" | grep -e bye
echo $?
if echo "hello world" | grep -e hello;then
echo true
else
echo false
fi
if echo "hello world" | grep -e bye;then
echo true
else
echo false
fi
demoFun1(){
return 0
}
demoFun2(){
return 12
}
if demoFun1;then
echo true
else
echo false
fi
if demoFun2;then
echo true
else
echo false
fi
:<<EOF
#41.输入输出重定向
Shell 输入/输出重定向
大多数 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)
输入重定向:这样,本来需要从键盘获取输入的命令会转移到文件读取内容。
注意:输出重定向是大于号(>),输入重定向是小于号(<)。
EOF
echo "######输出重定向内容######"
echo "hello beautiful sky blue tears......" > tears.txt
echo "okay, this is rigth in life...at fact" >> tears.txt
echo "######输入重定向######"
wc -l < tears.txt
wc -l tears.txt
wc -l < tears.txt > t.txt #同时替换输入和输出,执行command1,从文件infile读取内容,然后将输出写入到outfile中
:<<EOF
#42.重定向深入讲解
一般情况下,每个 Unix/Linux 命令运行时都会打开三个文件:
0 标准输入文件(stdin):stdin的文件描述符为0,Unix程序默认从stdin读取数据。
1 标准输出文件(stdout):stdout 的文件描述符为1,Unix程序默认向stdout输出数据。
2 标准错误文件(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
如果希望对 stdin 和 stdout 都重定向,可以这样写:command < file1 >file2
command 命令将 stdin 重定向到 file1,将 stdout 重定向到 file2
EOF
:<<EOF
#43.Here Document
Here Document 是 Shell 中的一种特殊的重定向方式,用来将输入重定向到一个交互式 Shell 脚本或程序。
它的基本的形式如下:
command << delimiter
document
delimiter
它的作用是将两个 delimiter 之间的内容(document) 作为输入传递给 command。
注意:
结尾的delimiter 一定要顶格写,前面不能有任何字符,后面也不能有任何字符,包括空格和 tab 缩进。
开始的delimiter前后的空格会被忽略掉。
EOF
echo "###here document###"
#在命令行中通过 wc -l 命令计算 Here Document 的行数:
wc -l << EOF
hello
baby
blue sky
is good
ok
EOF
:<<!
我们也可以将 Here Document 用在脚本中,例如:
#!/bin/bash
# author
# url:www.rob.com
cat << EOF
欢迎来到
dadadada
www.rob.com
EOF
执行以上脚本,输出结果:
欢迎来到
dadadada
www.rob.com
!
:<<EOF
#44./dev/null 文件
如果希望执行某个命令,但又不希望在屏幕上显示输出结果,那么可以将输出重定向到 /dev/null:command > /dev/null
/dev/null 是一个特殊的文件,写入到它的内容都会被丢弃;如果尝试从该文件读取内容,那么什么也读不到。
但是 /dev/null 文件非常有用,将命令的输出重定向到它,会起到"禁止输出"的效果
如果希望屏蔽 stdout 和 stderr,可以这样写:command > /dev/null 2>&1
注意:0 是标准输入(STDIN),1 是标准输出(STDOUT),2 是标准错误输出(STDERR)。
这里的 2 和 > 之间不可以有空格,2> 是一体的时候才表示错误输出。
EOF
echo "dddddddddd" > /dev/null 2>&1
:<<EOF
#45.文件包含
和其他语言一样,Shell 也可以包含外部脚本。这样可以很方便的封装一些公用的代码作为一个独立的文件。
Shell 文件包含的语法格式如下:
. filename # 注意点号(.)和文件名中间有一空格
或
source filename
注:被包含的文件 test1.sh 不需要可执行权限。
EOF
echo "###文件包含###"
. ./zll_0.sh
source ./zll_1.sh
echo "张磊年龄是$age 体重是$weight"
#还可以执行包含的文件
sh zll.sh
./zll.sh
a02.sh脚本 如下
内容摘要
034.break 命令
035.continue语句
036.shell函数
037.return 语句
038.函数参数
039.$?仅对其上一条指令负责,一旦函数返回后其返回值没有立即保存入参数,那么其返回值将不再能通过 $? 获得
040.函数与命令的执行结果运用
041.输入输出重定向
042.重定向深入讲解
043.Here Document
044./dev/null 文件
045.文件包含
shell扫盲总结文档
1.了解shell及基本变量
【注】本内容大部分理论来之https://www.runoob.com/linux/linux-shell.html,推荐阅读原网站文章
1.1 什么是shell?
Shell 是一个用 C 语言编写的程序,它是用户使用 Linux 的桥梁。Shell 既是一种命令语言,又是一种程序设计语言。
Shell 是指一种应用程序,这个应用程序提供了一个界面,用户通过这个界面访问操作系统内核的服务。
Ken Thompson 的 sh 是第一种 Unix Shell,Windows Explorer 是一个典型的图形界面 Shell。
Shell 脚本(shell script),是一种为 shell 编写的脚本程序。
业界所说的 shell 通常都是指 shell 脚本,但读者朋友要知道,shell 和 shell script 是两个不同的概念。
由于习惯的原因,简洁起见,本文出现的 "shell编程" 都是指 shell 脚本编程,不是指开发 shell 自身
Shell 编程跟 JavaScript、php 编程一样,只要有一个能编写代码的文本编辑器和一个能解释执行的脚本解释器就可以了。
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,也就是 Bourne Again Shell,由于易用和免费,Bash 在日常工作中被广泛使用。同时,Bash 也是大多数Linux 系统默认的 Shell。
在一般情况下,人们并不区分 Bourne Shell 和 Bourne Again Shell,所以,像 #!/bin/sh,它同样也可以改为 #!/bin/bash。
#! 告诉系统其后路径所指定的程序即是解释此脚本文件的 Shell 程序。
打开文本编辑器(可以使用 vi/vim 命令来创建文件),新建一个文件 test.sh,扩展名为 sh(sh代表shell),扩展名并不影响脚本执行,见名知意就好,如果你用 php 写 shell 脚本,扩展名就用 php 好了。
#! 是一个约定的标记,它告诉系统这个脚本需要什么解释器来执行,即使用哪一种 Shell。
echo 命令用于向窗口输出文本
2、作为解释器参数
这种运行方式是,直接运行解释器,其参数就是 shell 脚本的文件名,如:
/bin/sh test.sh
/bin/php test.php
这种方式运行的脚本,不需要在第一行指定解释器信息,写了也没用。
1.2 定义和使用变量
定义变量:
cur=1
name=zll
str=”body”
使用变量:
echo $cur
1.3 变量类型
运行shell时,会同时存在三种变量:
- 1) 局部变量 局部变量在脚本或命令中定义,仅在当前shell实例中有效,其他shell启动的程序不能访问局部变量。
- 2) 环境变量 所有的程序,包括shell启动的程序,都能访问环境变量,有些程序需要环境变量来保证其正常运行。必要的时候shell脚本也可以定义环境变量。
- 3) shell变量 shell变量是由shell程序设置的特殊变量。shell变量中有一部分是环境变量,有一部分是局部变量,这些变量保证了shell的正常运行
#!/bin/bash
#获取当前路径
CURRENT_PATH=$(pwd)
echo "$CURRENT_PATH"
变量类型
运行shell时,会同时存在三种变量:
1) 局部变量 局部变量在脚本或命令中定义,仅在当前shell实例中有效,其他shell启动的程序不能访问局部变量。
2) 环境变量 所有的程序,包括shell启动的程序,都能访问环境变量,有些程序需要环境变量来保证其正常运行。必要的时候shell脚本也可以定义环境变量。
3) shell变量 shell变量是由shell程序设置的特殊变量。shell变量中有一部分是环境变量,有一部分是局部变量,这些变量保证了shell的正常运行
1.4 字符串
字符串是shell编程中最常用最有用的数据类型(除了数字和字符串,也没啥其它类型好用了),字符串可以用单引号,也可以用双引号,也可以不用引号
单引号字符串的限制:
- 单引号里的任何字符都会原样输出,单引号字符串中的变量是无效的;
- 单引号字串中不能出现单独一个的单引号(对单引号使用转义符后也不行),但可成对出现,作为字符串拼接使用
双引号的优点:
- 双引号里可以有变量
- 双引号里可以出现转义字符
2.流程控制
2.1 if语句
语法格式:
if condition
then
command1
command2
...
commandN
fi
写成一行(适用于终端命令提示符):
if [ $(ps -ef | grep -c "ssh") -gt 1 ]; then echo "true"; fi
末尾的 fi 就是 if 倒过来拼写,后面还会遇到类似的。
if else 语法格式:
if condition
then
command1
command2
...
commandN
else
command
fi
if else-if else 语法格式:
if condition1
then
command1
elif condition2
then
command2
else
commandN
fi
if else 的 [...] 判断语句中大于使用 -gt,小于使用 -lt。
if [ "$a" -gt "$b" ]; then
...
fi
如果使用 ((...)) 作为判断语句,大于和小于可以直接使用 > 和 <
if (( a > b )); then
...
fi
################################################################
#029.if语句
#
#语法格式1:
#if [ ];then
# ...
#else
# ...
#fi
#
#语法格式2:
#if [ ];then
# ...
#elif [ ];then
# ...
#elif [ ];then
# ...
#else
# ...
#fi
#
#语法格式3:
#if [ ]
#then
# ...
#elif [ ]
#then
# ...
#else
# ...
#fi
#
#语法格式4:
#if [ ]
#then
# ...
#else
# ...
#fi
#
#语法格式5:
#if test -z $strs0;then
# ...
#else
# ...
#fi
#
#语法格式6:
#if (( a > b ));then
# ...
#elif (( a == b ));then
# ...
#else
# ...
#fi
#如果使用 ((...)) 作为判断语句,大于和小于可以直接使用 > 和 <
#
#if (( a > b )); then
# ...
#fi
################################################################
###############
#029.if语句
##############
echo "###if语句测试###"
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
str0=root
str1=testings
str2=God
#比较串是否一样, USER表示当前用户
if [ $USER == "$str0" ];then
echo "welcome root......"
else
echo "Sorry, you aren't root!"
fi
#比较字符串大小:大于、小于号必须加转移符号,否则会被认为是重定向 \> \<
if [ "$str1" \> "$str2" ];then
echo "$str1 大于 $str2"
else
echo "$str1 小于 $str2"
fi
#判断字符串长度是否非零:非零则为真,零则为假
if [ -n "$str3" ];then
echo "$str3 is not EMPTY!!"
else
echo "$str3 is EMPTY"
fi
#判断字符串是否为零:零则为真,非零则为假
if [ -z "$str3" ];then
echo "str3 is Zero"
else
echo "str3 is not Zero"
fi
2.2 for循环语句
####################################################
#030.for语句
#
#语法格式1:
#for doo in 1 2 3 4;do
# ...
#done
#
#语法格式2:
#for doo in 1 2 3 4
#do
# ...
#done
###################################################
#############
#030.for语句
#############
echo "###for循环###"
#循环遍历输出数字
for cur in 1 2 3 4 0
do
echo "The value is $cur"
done
#循环遍历输出字符串
for driver in This is a dog
do
echo "This value is $driver"
done
#嵌套if语句的for循环
CURRENT_PATH=$(pwd)
one="${CURRENT_PATH}/zll.sh"
DRIVER=hostapd.wlan0
for doo in $DRIVER; do (
if [ -x $one ];then
echo "top.sh is x file"
sh $one
else
echo "top.sh isn't x file"
fi
);done
2.3 while循环语句
:<<EOF
while 循环用于不断执行一系列命令,也用于从输入文件中读取数据。其语法格式为:
while condition;do
command
done
EOF
echo "###while循环测试###"
int=0
while(( int < 5 ))
do
echo $int
let "int++"
done
i=0
name="thing..."
while((i < 5));do
echo $name
let "i++"
done
2.4 untuil循环
:<<EOF
032.until循环
until 循环执行一系列命令直至条件为 true 时停止。
until 循环与 while 循环在处理方式上刚好相反。
一般 while 循环优于 until 循环,但在某些时候—也只是极少数情况下,until 循环更加有用。
until 语法格式:
until condition;do
command
done
注:condition 一般为条件表达式,如果返回值为 false,则继续执行循环体内的语句,否则跳出循环
EOF
ar=0
echo "###until循环###"
until (($ar > 10));do
echo $ar
ar=`expr $ar + 1`
done
2.5 case...esac语句
:<<EOF
033.case语句
case ... esac 为多选择语句,与其他语言中的 switch ... case 语句类似,是一种多分支选择结构,
每个 case 分支用右圆括号开始,用两个分号 ;; 表示 break,即执行结束,跳出整个 case ... esac 语句,
esac(就是 case 反过来)作为结束标记。
可以用 case 语句匹配一个值与一个模式,如果匹配成功,执行相匹配的命令。
语法格式:
case 值 in
模式1)
command1
command2
...
commandN
;;
模式2)
command1
command2
...
commandN
;;
esac
case 工作方式如上所示,取值后面必须为单词 in,每一模式必须以右括号结束。
取值可以为变量或常数,匹配发现取值符合某一模式后,其间所有命令开始执行直至 ;;。
取值将检测匹配的每一个模式。一旦模式匹配,则执行完匹配模式相应命令后不再继续其他模式。
如果无一匹配模式,使用星号 * 捕获该值,再执行后面的命令
EOF
echo "###case语句###"
read ay
case $ay in
1)echo "你写了1"
echo it is good
;;
2)echo "你写了2"
echo it is good
;;
3)echo "你写了3"
echo it is good
;;
4)echo "你写了4"
echo it is good
;;
*)echo "你写的数字不在1-4之内"
echo it is not good
;;
esac
2.6 break语句
:<<EOF
在循环过程中,有时候需要在未达到循环结束条件时强制跳出循环,Shell 使用两个命令来实现该功能:break 和 continue。
034.break 命令
break 命令允许跳出所有循环(终止执行后面的所有循环)。
EOF
echo "###break语句###"
while true;do
echo "输入1-5之间的数字"
read num
case $num in
1|2|3|4|5)echo "你输入的数字是$num"
echo "请继续..."
;;
*)echo "你输入的不是1-5"
echo "goodbye"
break
;;
esac
done
2.6 continue语句
#035.continue语句
#continue 命令与 break 命令类似,只有一点差别,它不会跳出所有循环,仅仅跳出当前循环
#运行代码发现,当输入大于5的数字时,该例中的循环不会结束,语句 echo "游戏结束" 永远不会被执行。
echo "###continue语句###"
while true;do
echo "输入1-5之间的数字"
read num
case $num in
1|2|3|4|5)echo "你输入的数字是$num"
echo "请继续..."
;;
*)echo "你输入的不是1-5"
echo "goodbye"
continue
echo "游戏结束"
;;
esac
done
3.运算符
Shell 和其他编程语言一样,支持多种运算符,包括:
- 算数运算符
- 关系运算符
- 布尔运算符
- 字符串运算符
- 文件测试运算符
原生bash不支持简单的数学运算,但是可以通过其他命令来实现,例如 awk 和 expr,expr 最常用。
expr 是一款表达式计算工具,使用它能完成表达式的求值操作。
例如,两个数相加(注意使用的是反引号 ` 而不是单引号 '):
#!/bin/bash
val=`expr 2 + 2`
echo "两数之和为 : $val"
两点注意:
- 表达式和运算符之间要有空格,例如 2+2 是不对的,必须写成 2 + 2,这与我们熟悉的大多数编程语言不一样。
完整的表达式要被 ` ` 包含,注意这个字符不是常用的单引号,在 Esc 键下边
3.1算术运算符
#019.算数运算符
hk=10
wk=30 #赋值
echo "#######算数运算符######演示######"
ak=`expr $hk + $wk` #加
echo "ak-------$ak"
bk=`expr $hk - $wk` #减
echo "bk-------$bk"
ck=`expr $hk \* $wk` #乘
echo "ck-------$ck"
dk=`expr $hk / $wk` #除
echo "dk-------$dk"
ek=`expr $hk % $wk` #取余
echo "ek-------$ek"
if [ $hk == $wk ];then
echo "hk 等于 wk"
fi
if [ $hk != $wk ];then
echo "hk 不等于 wk"
fi
3.2关系运算符
readonly aa=10
readonly bb=20
echo "===关系运算符==="
if [ $aa -eq $bb ];then
echo "aa等于bb"
fi
if [ $aa -ne $bb ];then
echo "aa不等于bb"
fi
if [ $aa -gt $bb ];then
echo "aa大于bb"
fi
if [ $aa -lt $bb ];then
echo "aa小于bb"
fi
if [ $aa -ge $bb ];then
echo "aa大于等于bb"
fi
[ $aa -le $bb ] && {
echo "aa小于等于bb"
}
3.3 布尔运算符
readonly ab=10
readonly ac=20
readonly ad=0
echo "=====布尔运算符======"
[ $ab ] && {
echo "-----here_0-------"
}
[ !$ad ] && {
echo "-----here_1-------"
}
[ $ab -lt $ac -o $ab -eq $ac ] && {
echo "------here_2------"
}
[ $ab -eq $ac -a $ab -gt $ac ] && {
echo "-----here_3-------"
}
3.4逻辑运算符
readonly ae=10
readonly af=20
echo "#####逻辑运算符#####"
if [[ $ae -lt 50 && $af -ge 20 ]];then
echo "ae小于50且af大于等于20"
fi
[ $ae -eq 10 ] && {
echo "ae 等于10"
}
if [[ $ae -gt 30 || af -lt 50 ]];then
echo "ae大于30或者af小于50"
fi
3.5字符串运算符
readonly strs0="abcdef"
readonly strs1="cdefgh"
readonly strs2="abcdef"
readonly strs3=
echo "###字符串运算###"
if [ $strs0 = $strs2 ];then
echo "strs0 = strs2"
fi
[ $strs0 != $strs1 ] && {
echo "strs0 != strs1"
}
if [ -z $strs3 ];then
echo "-z字符串strs3长度为零"
fi
if [ -n $strs0 ];then
echo "-n字符串strs0长度不为零"
fi
if [ $strs0 ];then
echo "字符串strs0不为空 $"
fi
3.6文件测试运算符
CURRENT_PATH=$(pwd)
file0="${CURRENT_PATH}/ltop"
file2="${CURRENT_PATH}/top.sh"
echo "###检测文件运算符###"
if [ -b $file0 ];then
echo "$file0 是块设备"
else
echo "$file0 不是块设备"
fi
if [ -d $file0 ];then
echo "$file0 是目录"
else
echo "$file0 不是目录"
fi
if [ -f $file2 ];then
echo "$file2 是普通文件"
else
echo "$file2 不是普通文件,是目录或者设备文件"
fi
if [ -L $file0 ];then
echo "$file0 是一个符号链接"
else
echo "$file0 不是一个符号链接"
fi
if [ -x $file2 ];then
echo "$file2 具有可执行权限"
else
echo "$file2 不具有可执行权限"
fi
if [ -e $file2 ];then
echo "$file2 存在此文件"
else
echo "$file2 不存在此文件"
fi
4.echo命令
#025.echo命令
readonly cmd="string is long"
echo "###echo命令###"
#输出字符
echo $cmd
echo cmd
echo "it is good thing"
echo it is good thing
#输出转移字符
echo "\"it is good thing\""
#显示变量
#read 命令从标准输入中读取一行,并把输入行的每个字段的值指定给 shell 变量
read bot #运行后会停留,让你输入bot的值,然后再运行完毕
echo "$bot is here"
#显示换行
echo -e "OK! \n"
#显示不换行
echo -e "OKK! \c"
echo "string is heer"
#输出结果定向到文件
echo "he is a boy" >my.txt
#原样输出字符串,不进行转义或取变量(用单引号)
echo '$cmd\"'
echo '$cmd'
#显示命令执行结果
echo `date`
5.printf命令
printf 命令模仿 C 程序库(library)里的 printf() 程序。
printf 由 POSIX 标准所定义,因此使用 printf 的脚本比使用 echo 移植性好。
printf 使用引用文本或空格分隔的参数,外面可以在 printf 中使用格式化字符串,还可以制定字符串的宽度、左右对齐方式等。默认的 printf 不会像 echo 自动添加换行符,我们可以手动添加 \n。
printf 命令的语法:
printf format-string [arguments...]
参数说明:
- format-string: 为格式控制字符串
arguments: 为参数列表。
echo "###printf命令###"
echo "hello world"
printf "hello world\n"
printf "%-10s %-8s %-4s\n" 姓名 性别 体重kg
printf "%-10s %-8s %-4.2f\n" 张三 男 80.1223
printf "%-10s %-8s %-4.2f\n" 李四 男 56.8945
printf "%-10s %-8s %-4.2f\n" 王五 男 39.6776
printf "%d %s\n" 10 "astring"
printf '%d %s\n' 10 "astring"
printf %s abcdefg
#格式只指定了一个参数,但多出的参数仍然会按照该格式输出,format-string 被重用
printf %s hhh good this
#如果没有 arguments,那么 %s 用NULL代替,%d 用 0 代替
printf "%s and %d \n"
6.Shell test 命令
6.1数值测试
readonly num_0=100
readonly num_1=200
echo "###test数值测试命令###"
if test $[num_0] -lt $[num_1];then
echo "$num_0小于$num_1"
else
echo "$num_0大于$num_1"
fi
#注意:代码中的 [] 执行基本的算数运算,等号两边不能有空格,如下:
result=$[num_0*num_1]
echo "result的值为:$result"
6.2字符串测试
echo "###test字符串测试命令###"
readonly strs0="jackoff"
readonly strs1="jackon"
if test $strs0 = $strs1;then
echo "两个字符串相等"
else
echo "两个字符串不相等"
fi
if test -n $strs0;then
echo "字符串长度不为零"
else
echo "字符串长度为零"
fi
6.3文件测试
CURRENT_PATH=$(pwd)
file0="${CURRENT_PATH}/top.sh"
file1="${CURRENT_PATH}/ltop"
echo "###test文件测试命令###"
if test -e $file0;then
echo "文件存在"
else
echo "文件不存在"
fi
if test -w $file1;then
echo "文件具有写权限"
else
echo "文件没有写权限"
fi
7.Shell 函数
7.1 不带返回值函数的使用
:<<EOF
036.shell函数
linux shell 可以用户定义函数,然后在shell脚本中可以随便调用。
shell中函数的定义格式如下:
[ function ] funname [()]
{
action;
[return int;]
}
说明:
1、可以带function fun() 定义,也可以直接fun() 定义,不带任何参数。
2、参数返回,可以显示加:return 返回,如果不加,将以最后一条命令运行结果,作为返回值。
return后跟数值n(0-255)
EOF
echo "###shell函数###"
demofun(){
echo "这是一个没什么用的函数吧"
}
echo "调用函数demofun"
demofun
echo "调用完毕"
7.2 带返回值函数的使用
:<<EOF
函数返回值在调用该函数后通过 $? 来获得。
注意:所有函数在使用前必须定义。这意味着必须将函数放在脚本开始部分,
直至shell解释器首次发现它时,才可以使用。调用函数仅使用其函数名即可
EOF
echo "###带返回值的函数使用###"
addnum(){
echo "求和运算并返回和值"
echo "输入第一个参数是:"
read Firstnum
echo "输入第二个参数是:"
read Secondnum
echo "输入的两个参数分别是:$Firstnum 和 $Secondnum"
echo "两数之和是: $(($Firstnum+$Secondnum))"
return $(($Firstnum+$Secondnum))
}
addnum
echo "函数的返回值是 $?"
7.3 函数参数
:<<EOF
38.函数参数
在Shell中,调用函数时可以向其传递参数。在函数体内部,
通过 $n 的形式来获取参数的值,例如,$1表示第一个参数,$2表示第二个参数...
注意,$10 不能获取第十个参数,获取第十个参数需要${10}。当n>=10时,需要使用${n}来获取参数。
另外,还有几个特殊字符用来处理参数:
参数处理 说明
$# 传递到脚本或函数的参数个数
$* 以一个单字符串显示所有向脚本传递的参数
$$ 脚本运行的当前进程ID号
$! 后台运行的最后一个进程的ID号
$@ 与$*相同,但是使用时加引号,并在引号中返回每个参数。
$- 显示Shell使用的当前选项,与set命令功能相同。
$? 显示最后命令的退出状态。0表示没有错误,其他任何值表明有错误。
EOF
echo "###函数参数运用###"
printf_para(){
echo "输入第1个参数是:$1"
echo "输入第5个参数是:$5"
echo "输入第9个参数是:$9"
echo "输入第10个参数是:$10"
echo "输入第10个参数是:${10}"
echo "输入第11个参数是:${11}"
echo "输入第11个参数是:$11"
echo "输入参数总个数为:$#个"
echo "输入的参数是:$*"
echo "输入的参数是:$@"
}
printf_para 1 2 3 4 5 6 7 8 9 10 11 12 16 18 19 20
7.4 $?的作用域
#39.$?仅对其上一条指令负责,一旦函数返回后其返回值没有立即保存入参数,那么其返回值将不再能通过 $? 获得
echo "###$? 作用效果域演示###"
function addnumber(){
echo "这个还是加法运算..."
num_0=9
num_1=10
sk=`expr $num_0 + $num_1`
return $sk
}
addnumber
echo $?
echo "加入一个语句..."
echo $?
7.5 函数与命令的执行结果作为条件语句使用
:<<EOF
#40.函数与命令的执行结果运用
函数与命令的执行结果可以作为条件语句使用。要注意的是,和 C 语言不同,
shell 语言中 0 代表 true,0 以外的值代表 false
如下:
grep 是从给定字符串中寻找匹配内容的命令。首先看出如果找到了匹配的内容,
会打印匹配部分且得到的返回值 $? 为 0,如果找不到,则返回值 $? 为 1
接下来分别将这两次执行的 grep 命令当作条件语句交给 if 判断,
得出返回值 $? 为 0,即执行成功时,条件语句为 true,当返回值 $? 为 1,即执行失败时,条件语句为 false。
之后再用函数的 return 值作为测试,其中 demoFun1 返回值为 0,
demoFun2 返回值选择了任意一个和 0 不同的整数,这里为 12。
将函数作为条件语句交给 if 判断,得出返回值为 0 时,依然为 true,
而返回值只要不是 0,条件语句都判断为 false
EOF
echo "###函数与命令的执行结果作为条件语句使用###"
echo "hello world" | grep -e hello
echo $?
echo "hello world" | grep -e bye
echo $?
if echo "hello world" | grep -e hello;then
echo true
else
echo false
fi
if echo "hello world" | grep -e bye;then
echo true
else
echo false
fi
demoFun1(){
return 0
}
demoFun2(){
return 12
}
if demoFun1;then
echo true
else
echo false
fi
if demoFun2;then
echo true
else
echo false
fi
8.重定向
:<<EOF
#41.输入输出重定向
Shell 输入/输出重定向
大多数 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)
输入重定向:这样,本来需要从键盘获取输入的命令会转移到文件读取内容。
注意:输出重定向是大于号(>),输入重定向是小于号(<)。
EOF
echo "######输出重定向内容######"
echo "hello beautiful sky blue tears......" > tears.txt
echo "okay, this is rigth in life...at fact" >> tears.txt
echo "######输入重定向######"
wc -l < tears.txt
wc -l tears.txt
wc -l < tears.txt > t.txt #同时替换输入和输出,执行command1,从文件infile读取内容,然后将输出写入到outfile中
:<<EOF
#42.重定向深入讲解
一般情况下,每个 Unix/Linux 命令运行时都会打开三个文件:
0 标准输入文件(stdin):stdin的文件描述符为0,Unix程序默认从stdin读取数据。
1 标准输出文件(stdout):stdout 的文件描述符为1,Unix程序默认向stdout输出数据。
2 标准错误文件(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
如果希望对 stdin 和 stdout 都重定向,可以这样写:command < file1 >file2
command 命令将 stdin 重定向到 file1,将 stdout 重定向到 file2
EOF
:<<EOF
#43.Here Document
Here Document 是 Shell 中的一种特殊的重定向方式,用来将输入重定向到一个交互式 Shell 脚本或程序。
它的基本的形式如下:
command << delimiter
document
delimiter
它的作用是将两个 delimiter 之间的内容(document) 作为输入传递给 command。
注意:
结尾的delimiter 一定要顶格写,前面不能有任何字符,后面也不能有任何字符,包括空格和 tab 缩进。
开始的delimiter前后的空格会被忽略掉。
EOF
echo "###here document###"
#在命令行中通过 wc -l 命令计算 Here Document 的行数:
wc -l << EOF
hello
baby
blue sky
is good
ok
EOF
:<<!
我们也可以将 Here Document 用在脚本中,例如:
#!/bin/bash
# author
# url:www.rob.com
cat << EOF
欢迎来到
dadadada
www.rob.com
EOF
执行以上脚本,输出结果:
欢迎来到
dadadada
www.rob.com
!
:<<EOF
#44./dev/null 文件
如果希望执行某个命令,但又不希望在屏幕上显示输出结果,那么可以将输出重定向到 /dev/null:command > /dev/null
/dev/null 是一个特殊的文件,写入到它的内容都会被丢弃;如果尝试从该文件读取内容,那么什么也读不到。
但是 /dev/null 文件非常有用,将命令的输出重定向到它,会起到"禁止输出"的效果
如果希望屏蔽 stdout 和 stderr,可以这样写:command > /dev/null 2>&1
注意:0 是标准输入(STDIN),1 是标准输出(STDOUT),2 是标准错误输出(STDERR)。
这里的 2 和 > 之间不可以有空格,2> 是一体的时候才表示错误输出。
EOF
echo "dddddddddd" > /dev/null 2>&1
9.文件包含
:<<EOF
#45.文件包含
和其他语言一样,Shell 也可以包含外部脚本。这样可以很方便的封装一些公用的代码作为一个独立的文件。
Shell 文件包含的语法格式如下:
. filename # 注意点号(.)和文件名中间有一空格
或
source filename
注:被包含的文件 test1.sh 不需要可执行权限。
EOF
echo "###文件包含###"
. ./zll_0.sh
source ./zll_1.sh
echo "张磊年龄是$age 体重是$weight"