概论
shell是我们通过命令行与操作系统沟通的语言。
shell脚本可以直接在命令行中执行,也可以将一套逻辑组织成一个文件,方便复用。
AC Terminal中的命令行可以看成是一个“shell脚本在逐行执行”。
Linux中常见的shell脚本有很多种,常见的有:
-
Bourne Shell(/usr/bin/sh或/bin/sh)
-
Bourne Again Shell(/bin/bash)
-
C Shell(/usr/bin/csh)
-
K Shell(/usr/bin/ksh)
-
zsh
Linux系统中一般默认使用bash,所以接下来讲解bash中的语法。
文件开头需要写**#! /bin/bash
**,指明bash
为脚本解释器。
脚本示例
新建一个test.sh
文件,内容如下:
1 | ! /bin/bash |
运行方式
- 使脚本具有可执行权限
chmod +x test.sh
- 不同路径下运行
1 | ./test.sh #当前路径 |
注释
单行注释
每行中#
之后的内容均是注释。
1 | 这是一行注释 |
多行注释
1 | :<<EOF |
其中EOF可以换成其它任意字符串。例如:
1 | :<<abc |
echo 命令
echo
用于输出字符串。命令格式:echo string
显示普通字符串
1 | echo "Hello BrownCutie" |
显示转义字符
1 | echo "\"Hello BrownCutie\"" # 注意只能使用双引号,如果使用单引号,则不转义 |
显示变量
1 | name=brown |
显示换行
1 | echo -e "Hi\n" # -e 开启转义 |
结果
1 | Hi |
不显示换行
1 | echo -e "Hi \c" # -e 开启转义 \c 不换行 |
结果
1 | Hi brown |
显示结果定向至文件
1 | echo "Hello World" > output.txt # 将内容以覆盖的方式输出到output.txt中 |
原样输出字符串,不进行转义或取变量(用单引号)
1 | name=brown |
结果 $name\"
显示命令的执行结果
1 | echo `date` |
结果
1 | Wed Feb 15 04:38:51 CST 2023 |
printf命令
printf
命令用于格式化输出,类似于C/C++
中的printf
函数。
默认不会在字符串末尾添加换行符。
命令格式:
printf format-string [arguments...]
用法示例
脚本内容:
1 | printf "%10d.\n" 123 # 占10位,右对齐 |
输出结果:
1 | 123. |
变量
定义变量
定义变量,不需要加$符号,例如:
1 | name1='brown' # 单引号定义字符串 |
使用变量
使用变量,需要加上$
符号,或者${}
符号。花括号是可选的,主要为了帮助解释器识别变量边界。
1 | name=brown |
只读变量
使用readonly
或者declare
可以将变量变为只读。
1 | name=yxc |
删除变量
unset
可以删除变量。
1 | name=yxc |
变量类型
-
自定义变量(局部变量)
子进程不能访问的变量 -
环境变量(全局变量)
子进程可以访问的变量
1 | 自定义变量改成环境变量 |
验证全局变量可以试一试下面的代码
1 | name=brown |
字符串
- 字符串可以用单引号,也可以用双引号,也可以不用引号。
单引号与双引号的区别:
- 单引号中的内容会原样输出,不会执行、不会取变量;
- 双引号中的内容可以执行、可以取变量;
1 | name=brown # 不用引号 |
- 获取字符串长度
1 | name="yxc" |
- 提取子串
${string:begin:len}
和c\c++
一样从begin
开始截取长度len
的字符串(索引从0开始)
1 | name="hello, brown" |
默认变量
- 文件参数变量
在执行shell
脚本时,可以向脚本传递参数。$1
是第一个参数,$2
是第二个参数,以此类推。特殊的,$0
是文件名(包含路径)。
其实$0
就是怎么调用的.sh
文件就打印什么,跟绝对路径还是相对路径调用的有关
创建文件test.sh:
1 | ! /bin/bash |
然后执行该脚本:
1 | chmod +x test.sh |
- 其它参数相关变量
参数 | 说明 |
---|---|
$# |
代表文件传入的参数个数,如上例中值为4 |
$* |
由所有参数构成的用空格隔开的字符串,如上例中值为"$1 $2 $3 $4" |
$@ |
每个参数分别用双引号括起来的字符串,如上例中值为"$1" "$2" "$3" "$4" |
$$ |
脚本当前运行的进程ID |
$? |
上一条命令的退出状态(注意不是stdout,而是exit code)。0表示正常退出,其他值表示错误 |
$(command) |
返回command这条命令的stdout(可嵌套) |
`command` | 返回command这条命令的stdout(不可嵌套) |
对于$?
的说明正常退出返回的,否则就是1~255
的一个数字来表示异常
对于以后的语法如果看的是exit code
那么就可以理解为真,其他为假
如果看的是stdout
则相反
数组
数组中可以存放多个不同类型的值,只支持一维数组,初始化时不需要指明数组大小。
数组下标从0开始。
定义
数组用小括号表示,元素之间用空格隔开。例如:
1 | array=(1 abc "def" yxc) |
也可以直接定义数组中某个元素的值:(二者等价)
1 | array[0]=1 |
除此之外第二种方法还可以不用按照顺序定义
1 | array[0]=1 |
上述这种定义方式并不会开个长度的数组,只会开实际长度四个长度.后续求数组长度也是,可以将其理解为一种hash
读取数组中某个元素的值
格式:${array[index]}
1 | array=(1 abc "def" yxc) |
读取整个数组
1 | array=(1 abc "def" brown) |
数组长度
类似于字符串
1 | array=(1 abc "def" brown) |
read命令
read
命令用于从标准输入中读取单行数据。当读到文件结束符时,exit code
为1
,否则为0
。
参数说明
-p
: 后面可以接提示信息
-t
:后面跟秒数,定义输入字符的等待时间,超过等待时间后会自动忽略此命令
实例:
1 | read name # 读入name的值 |
exit命令
exit
命令用来退出当前shell进程,并返回一个退出状态;使用$?
可以接收这个退出状态。
exit
命令可以接受一个整数值作为参数,代表退出状态。如果不指定,默认状态值是0
。
exit
退出状态只能是一个介于 0~255
之间的整数,其中只有0
表示成功,其它值都表示失败。
1 | ! /bin/bash |
执行一下
1 | chmod +x test.sh |
文件重定向
每个进程默认打开3个文件描述符:
stdin
标准输入,从命令行读取数据,文件描述符为0
stdout
标准输出,向命令行输出数据,文件描述符为1
stderr
标准错误输出,向命令行输出数据,文件描述符为2
可以用文件重定向将这三个文件重定向到其他文件中。
重定向命令列表
命令 | 说明 |
---|---|
command > file |
将stdout重定向到file中 |
command < file |
将stdin重定向到file中 |
command >> file |
将stdout以追加方式重定向到file中 |
command n> file |
将文件描述符n重定向到file中 |
command n>> file |
将文件描述符n以追加方式重定向到file中 |
输入和输出重定向
1 | echo -e "Hello \c" > output.txt # 将stdout重定向到output.txt中 |
同时重定向stdin和stdout
1 | ! /bin/bash |
创建input.txt
,里面的内容为:
1 | 3 |
执行
1 | chmod +x test.sh # 添加可执行权限 |
不能写成
input.txt > ./test.sh > output.txt
stdin
和stdout
顺序可以改变<
就是输入>
就是输出 下面几种都可以
1
2 < input.txt ./test.sh > output.txt
output.txt < input.txt ./test.sh
引入外部脚本
类似于C/C++
中的include
操作,bash
也可以引入其他文件中的代码。
语法格式:
1 | . filename # 注意点和文件名之间有一个空格 |
示例
创建test1.sh
,内容为:
1 | ! /bin/bash |
然后创建test2.sh
,内容为:
1 | ! /bin/bash |
执行命令:
1 | chmod +x test2.sh |