引用:

https://www.acwing.com/file_system/file/content/whole/index/content/2855883/

shell语法

概论

shell是我们通过命令行与操作系统沟通的语言。

shell脚本可以直接在命令行中执行,也可以将一套逻辑组织成一个文件,方便复用。
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
2
#! /bin/bash
echo "Hello World!"

运行方式

作为可执行文件

1
2
3
4
5
6
7
8
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! # 脚本输出

用解释器执行

1
2
acs@9e0ebfcd82d7:~$ bash test.sh
Hello World! # 脚本输出

注:用解释器执行不需要有权限,作为可执行文件来执行一定要有权限。

注释

单行注释

每行中#之后的内容均是注释。

1
2
# 这是一行注释
echo 'Hello World' # 这也是注释

多行注释

格式:

1
2
3
4
5
:<<EOF
第一行注释
第二行注释
第三行注释
EOF

其中EOF可以换成其它任意字符串。例如:

1
2
3
4
5
6
7
8
9
10
11
:<<abc
第一行注释
第二行注释
第三行注释
abc

:<<!
第一行注释
第二行注释
第三行注释
!

变量

定义变量

定义变量,不需要加$符号,例如:

1
2
3
name1='yxc'  # 单引号定义字符串
name2="yxc" # 双引号定义字符串
name3=yxc # 也可以不加引号,同样表示字符串

使用变量

使用变量,需要加上$符号,或者${}符号。花括号是可选的,主要为了帮助解释器识别变量边界。

1
2
3
4
name=yxc
echo $name # 输出yxc
echo ${name} # 输出yxc
echo ${name}acwing # 输出yxcacwing

只读变量

使用readonly或者declare可以将变量变为只读。

1
2
3
4
5
name=yxc
readonly name
declare -r name # 两种写法均可

name=abc # 会报错,因为此时name只读

删除变量

unset可以删除变量。

1
2
3
name=yxc
unset name
echo $name # 输出空行

变量类型

  1. 自定义变量(局部变量)
    子进程不能访问的变量

  2. 环境变量(全局变量)
    子进程可以访问的变量

    自定义变量改成环境变量:

1
2
3
acs@9e0ebfcd82d7:~$ name=yxc  # 定义变量
acs@9e0ebfcd82d7:~$ export name # 第一种方法
acs@9e0ebfcd82d7:~$ declare -x name # 第二种方法

环境变量改为自定义变量:

1
2
acs@9e0ebfcd82d7:~$ export name=yxc  # 定义环境变量
acs@9e0ebfcd82d7:~$ declare +x name # 改为自定义变量

字符串

字符串可以用单引号,也可以用双引号,也可以不用引号。

单引号与双引号的区别:

  • 单引号中的内容会原样输出,不会执行、不会取变量;
  • 双引号中的内容可以执行、可以取变量;
1
2
3
name=yxc  # 不用引号
echo 'hello, $name \"hh\"' # 单引号字符串,输出 hello, $name \"hh\"
echo "hello, $name \"hh\"" # 双引号字符串,输出 hello, yxc "hh"

获取字符串长度

1
2
name="yxc"
echo ${#name} # 输出3

提取子串

1
2
name="hello, yxc"
echo ${name:0:5} # 提取从0开始的5个字符

默认变量

文件参数变量

在执行shell脚本时,可以向脚本传递参数。$1是第一个参数,$2是第二个参数,以此类推。特殊的,$0是文件名(包括路径)。列如:

创建文件test.sh

1
2
3
4
5
6
7
#! /bin/bash

echo "文件名:"$0
echo "第一个参数:"$1
echo "第二个参数:"$2
echo "第三个参数:"$3
echo "第四个参数:"$4

然后执行脚本

1
2
3
4
5
6
7
chmod +x test.sh
./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(不可嵌套)

数组

定义

数组用小括号表示,元素之间用空格隔开。列如:

1
array=(1 abc "def" lg)

也可以直接定义数组中某个元素的值

1
2
3
4
array[0]=1
array[1]=abc
array[2]=def
array[3]=lg

读取数组中某个元素的值

格式

1
${array[index]}

例如:

1
2
3
4
5
array=(1 abc "def" lg)
echo ${array[0]}
echo ${array[1]}
echo ${array[2]}
echo ${array[3]}

读取整个数组

格式:

1
2
${array[@]} # 第一种写法
${array[*]} # 第二种写法

例如:

1
2
3
array=(1 abc "def" lg)
${array[@]} # 第一种写法
${array[*]} # 第二种写法