概论
shell是什么
-
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为脚本解释器。
学习技巧
- 不要死记硬背,遇到含糊不清的地方,可以在AC Terminal里实际运行一遍。
脚本示例
- 新建一个test.sh文件,内容如下:
#! /bin/bash
echo "Hello World!"
运行方式
作为可执行文件
# 作为可执行文件执行需要权限
acs@9e0ebfcd82d7:~$ ls -l test.sh # 查看权限
acs@9e0ebfcd82d7:~$ chmod +x test.sh # 使脚本具有可执行权限
acs@9e0ebfcd82d7:~$ ./test.sh # 当前路径下执行
Hello World! # 脚本输出
acs@9e0ebfcd82d7:~$ /home/acs/test.sh # 绝对路径下执行
Hello World! # 脚本输出
acs@9e0ebfcd82d7:~$ ~/test.sh # 家目录路径下执行
Hello World! # 脚本输出
用解释器执行
# 解释器执行不需要有可执行权限
acs@9e0ebfcd82d7:~$ bash test.sh
Hello World! # 脚本输出
注释
单行注释
- 每行中
#
之后的内容均是注释。
# 这是一行注释
echo 'Hello World' # 这也是注释
多行注释
# 格式:
:<<EOF
第一行注释
第二行注释
第三行注释
EOF
# 其中EOF可以换成其它任意字符串。例如:
:<<abc
第一行注释
第二行注释
第三行注释
abc
:<<!
第一行注释
第二行注释
第三行注释
!
变量
定义变量
- 定义变量,不需要加
$
符号,例如:
# 定义变量时,等号两边不能有空格
name1='hf' # 单引号定义字符串
name2="hf" # 双引号定义字符串
name3=hf # 也可以不加引号,同样表示字符串
使用变量
- 使用变量,需要加上
$
符号,或者${}
符号。 - 花括号是可选的,主要为了帮助解释器识别变量边界。
name=hf
echo $name # 输出yxc
echo ${name} # 输出yxc
echo ${name}acwing # 输出yxcacwing
只读变量
- 使用
readonly
或者declare
可以将变量变为只读。
name=hf
readonly name
declare -r name # 两种写法均可
name=abc # 会报错,因为此时name只读
删除变量
- unset可以删除变量。
# 被声明为只读的变量无法被unset删除
name=hf
unset name
echo $name # 输出空行
变量类型
-
自定义变量(局部变量):子进程不能访问的变量
-
环境变量(全局变量):子进程可以访问的变量
# 自定义变量改成环境变量:
# 默认变量是局部变量(自定义变量)
acs@9e0ebfcd82d7:~$ name=hf # 定义变量
acs@9e0ebfcd82d7:~$ export name # 第一种方法
acs@9e0ebfcd82d7:~$ declare -x name # 第二种方法
# 环境变量改为自定义变量:
acs@9e0ebfcd82d7:~$ export name=hf # 定义环境变量
acs@9e0ebfcd82d7:~$ declare +x name # 改为自定义变量
- tmux经
ctrl a + %
后分屏的两个界面实际上是两个bash,在一个bash中自定义的bash变量仅可以被当前bash访问,通过bash命令开启的子进程也无法访问。export后变量变为环境变量,就可以被全局访问。
字符串
-
字符串可以用单引号,也可以用双引号,也可以不用引号。
-
单引号与双引号的区别:
- 单引号中的内容会原样输出,不会执行、不会取变量;
- 双引号中的内容可以执行、可以取变量;
name=yxc # 不用引号
echo 'hello, $name \"hh\"' # 单引号字符串,输出 hello, $name \"hh\"
echo "hello, $name \"hh\"" # 双引号字符串,输出 hello, yxc "hh"
# 字符串中,不加引号和双引号效果相同
echo hello, $name \"hh\" # 无引号字符串,输出 hello, yxc "hh"
- 获取字符串长度
name="yxc"
echo ${#name} # 输出3
- 提取子串
name="hello, yxc"
echo ${name:0:5} # 提取从0开始的5个字符
默认变量
文件参数变量
-
在执行shell脚本时,可以向脚本传递参数。
$1
是第一个参数,$2
是第二个参数,以此类推。特殊的,$0
是文件名(包含路径)。例如: -
创建文件test.sh:
#! /bin/bash
echo "文件名:"$0
echo "第一个参数:"$1
echo "第二个参数:"$2
echo "第三个参数:"$3
echo "第四个参数:"$4
# 向脚本传递参数时,参数个数超过一位需要用大括号括起来
echo ${10}
echo ${11}
- 然后执行该脚本:
acs@9e0ebfcd82d7:~$ chmod +x test.sh
acs@9e0ebfcd82d7:~$ ./test.sh 1 2 3 4
文件名:./test.sh
第一个参数:1
第二个参数:2
第三个参数:3
第四个参数:4
- 其它参数相关变量
参数 | 说明 |
---|---|
$# |
代表文件传入的参数个数,如上例中值为4 |
$* |
由所有参数构成的用空格隔开的字符串,如上例中值为"$1 $2 $3 $4" |
$@ |
每个参数分别用双引号括起来的字符串,如上例中值为"$1" "$2" "$3" "$4" |
$$ |
脚本当前运行的进程ID |
$? |
上一条命令的退出状态(注意不是stdout,而是exit code,类似于返回值)。0表示正常退出,其他值表示错误 |
$(command) |
返回command 这条命令的stdout(可嵌套) |
command |
返回command 这条命令的stdout(不可嵌套) |
数组
- 数组中可以存放多个不同类型的值,只支持一维数组,初始化时不需要指明数组大小。
- 数组下标从0开始。
定义
- 数组用小括号表示,元素之间用空格隔开。例如:
array=(1 abc "def" yxc)
- 也可以直接定义数组中某个元素的值:
array[0]=1
array[1]=abc
array[2]="def"
array[3]=yxc
读取数组中某个元素的值
- 格式:
${array[index]}
- 例如:
array=(1 abc "def" yxc)
echo ${array[0]}
echo ${array[1]}
echo ${array[2]}
echo ${array[3]}
读取整个数组
- 格式:
${array[@]} # 第一种写法
${array[*]} # 第二种写法
- 例如:
array=(1 abc "def" yxc)
echo ${array[@]} # 第一种写法
echo ${array[*]} # 第二种写法
数组长度
- 类似于字符串
${#array[@]} # 第一种写法
${#array[*]} # 第二种写法
- 例如:
array=(1 abc "def" yxc)
echo ${#array[@]} # 第一种写法
echo ${#array[*]} # 第二种写法