Bash Shell脚本学习

Shell Script 是一种为 shell 编写的脚本程序。

学习Bash

  • Bash: Bourne Again Shell(/bin/bash)

完全参考:菜鸟笔记:Shell 教程,大部分是复制粘贴。

HelloWorld

1
2
#!/bin/bash
echo "Hello World !"
  • #! 脚本执行选用的解释器

  • echo 向窗口输出文本

  • 运行前,加上权限chmod + x /xx/xx/helloworld.sh

  • 运行方式1,./helloworld.sh ./表示告知 Bash sh文件在当前PATH

  • 运行方式2,/bin/bash helloworld.sh,这种方式不需要给出#!/bin/bash

Shell变量

定义变量

your_name=Ashin

  • =之间不能有空格
  • 变量命令如同C语言

使用变量

1
2
your_name="Ashin"
echo ${your_name}
  • {}帮助解释器识别变量的边界
  • 习惯加{ }

只读变量

使用readonly将变量设置为只读

1
2
3
4
5
6
7
8
ashin@MacBook shell % cat helloworld.sh 
#! /bin/bash
myUrl="https://ashin.wang"
readonly myUrl
myUrl="https://ashin.cc"

ashin@MacBook shell % ./helloworld.sh
./helloworld.sh: line 4: myUrl: readonly variable

删除变量

unset 变量名

实例变量

  • 局部变量

    定义在当前脚本或命令,仅当前的 Shell 有效。

  • 环境变量

    所有程序和 Shell 都能访问的 变量。

  • Shell变量

    Shell 程序设置的变量。

Shell字符串

单引号

str='string'

  • 单引号里的任何字符都会原样输出,单引号字符串中的变量是无效的。
  • 单引号需要成对出现,作为字符串拼接使用。

双引号

1
2
3
your_name="Ashin"
str="Hello, \"${your_name}\"!\n"
echo -e ${str}
1
2
ashin@MacBook shell % ./helloworld.sh  
Hello, "Ashin"!
  • 双引号中可以有变量

  • 双引号中可以有转义字符

拼接字符

  • 拼接
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
ashin@MacBook shell % cat helloworld.sh
#! /bin/bash
your_name="Ashin"
# 使用双引号拼接
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}

ashin@MacBook shell % ./helloworld.sh
Hello,Ashin! Hello,Ashin!
Hello,Ashin! Hello,${your_name}!

获取字符串长度

#string

1
2
3
4
5
6
ashin@MacBook shell % cat helloworld.sh 
#! /bin/bash
string="Ashin"
echo ${#string}
ashin@MacBook shell % ./helloworld.sh
5

提取字符串

1
2
3
4
string="Ashin"
echo ${string:0:4}

Ashi

Shell注释

  • 单行注释 #

  • 多行注释

    注释开头 :<<EOF:<<':<<! 注释结尾为注释开头最后一个字符

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
:<<EOF
注释内容...
注释内容...
注释内容...
EOF

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

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

Shell传递参数

1
2
3
4
5
6
7
8
9
10
11
12
13
#! /bin/bash
echo "Shell 传递参数实例"
echo "执行的文件名:$0"
echo "第一个参数为:$1"
echo "第二个参数为:$2"
echo "第三个参数为:$3"
# 执行脚本
ashin@MacBook shell % ./helloworld.sh a b c
Shell 传递参数实例
执行的文件名:./helloworld.sh
第一个参数为:a
第二个参数为:b
第三个参数为:c
参数处理 说明
$# 传递到脚本的参数个数
$* 以一个单字符串显示所有向脚本传递的参数。
$$ 脚本运行的当前进程ID号
$! 后台运行的最后一个进程的ID号
$@ $* 相同,但是使用时加引号,并在引号中返回每个参数。
$- 显示Shell使用的当前选项,与set功能相同。
$? 显示最后命令的退出状态。0表示没有错误,其他任何值表明有错误。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
ashin@MacBook shell % cat helloworld.sh 
#! /bin/bash
for i in "$*"; do
echo $i
done

for j in "$@"; do
echo $j
done

ashin@MacBook shell % ./helloworld.sh a b c
a b c
a
b
c

Shell数组

  • Bash 仅支持一维数组
  • 类似 C 语言,从 0 开始

定义数组

Arr_name=(value1 value2 value3 · · ·)

Arr_name[0]=value1

  • 空格分隔字符

读取数组

获取单个 ${Arr_name[n]}

获取全部 ${Arr_name[@]

获取数组长度

length=${Arr_name[@]

length=${Arr_name[*]}

  • 获取数组元素的长苏 length_n=${Arr_name[n]}

Shell运算符

  • Bash 不支持简单的数学运算,使用 expr

两数相加

1
2
3
4
5
6
7
ashin@MacBook shell % cat helloworld.sh 
#! /bin/bash
val=`expr 2 + 2`
echo "两数之和为:$val"

ashin@MacBook shell % ./helloworld.sh
两数之和为:4
  • 表达式和运算符之间要留空格,2 + 2

  • 完整的表达式需要使用 ` 包裹

算数运算符

  • 假设变量为 a, b
  • 条件表达式放在[ ]
运算符 说明 举例
+ 加法 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。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
ashin@MacBook shell % cat helloworld.sh 
#! /bin/bash
a=3
b=4
val=`expr $a \* $b`
echo "$val"

val=`expr $a / $b`
echo "$val"

if [ $a == $b ]
then
echo "a 等于 b"
fi
if [ $a != $b ]
then
echo "a 不等于 b"
fi
ashin@MacBook shell % ./helloworld.sh
12
0
a 不等于 b

关系运算符

运算符 说明 举例
-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。

布尔运算符

运算符 说明 举例
! 非运算,表达式为 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。

逻辑运算符

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

字符串运算符

运算符 说明 举例
= 检测两个字符串是否相等,相等返回 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。

文件测试运算符

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

echo命令

  • 用于字符串的输出,命令格式为echo string

  • \ 转义字符串

  • ""可要可不要,习惯使用

  • read 读入输入的第一行

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

  • echo "重定向" > 1.txt 重定向到文件1.txt

  • 使用 ` 包裹显示命令结果,例如: echo `date`

1
2
3
4
5
6
7
8
9
10
11
12
13
ashin@MacBook shell % echo hahaha 
hahaha
ashin@MacBook shell % echo "hahaha"
hahaha
ashin@MacBook shell % echo \"hahaha\"
"hahaha"
ashin@MacBook shell % cat helloworld.sh
#! /bin/bash
read name
echo "$name It is a test"
ashin@MacBook shell % ./helloworld.sh
halo
halo It is a test

printf命令

  • 类似 C 语言

流程控制

条件语句

if···fi

1
2
3
4
5
6
7
if condition
then
command1
command2
...
commandN
fi

if···else

1
2
3
4
5
6
7
8
9
if condition
then
command1
command2
...
commandN
else
command
fi

if···elif···else

1
2
3
4
5
6
7
8
9
if condition1
then
command1
elif condition2
then
command2
else
commandN
fi
  • 终端一行

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

循环语句

for循环

1
2
3
4
5
6
7
for var in item1 item2 ... itemN
do
command1
command2
...
commandN
done
  • 终端一行

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

while循环

1
2
3
4
while condition
do
command
done
1
2
3
4
5
6
7
8
9
10
11
12
13
14
ashin@MacBook shell % cat helloworld.sh 
#! /bin/bash
int=1
while(( $int<=5 ))
do
echo $int
let "int++"
done
ashin@MacBook shell % ./helloworld.sh
1
2
3
4
5

无线循环

1
2
3
4
while :
do
command
done
1
2
3
4
while true
do
command
done
1
for (( ; ; ))

until循环

until 循环直到 true 时停止,跟 while 相反

1
2
3
4
until condition
do
command
done

case

  • case 和 esac 相反包裹
1
2
3
4
5
6
7
8
9
10
11
12
13
14
case 值 in
模式1)
command1
command2
...
commandN
;;
模式2)
command1
command2
...
commandN
;;
esac

跳出循环

  • break 终止循环

  • continue 跳出本次循环

Shell函数

1
2
3
4
function_name(){

return int;
}

函数参数

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

输入、输出重定向

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

Shell文件引用

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