来自:Linux基础课
3.shell语法
文章目录
- 3.shell语法
- 3.1 概论
- 3.2 注释
- 3.3 变量
- 3.4 默认变量
- 3.5 数组
- 3.6 expr命令
- 3.7 read命令
- 3.8 echo命令
3.1 概论
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:~$ 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!
3.2 注释
单行注释
每行中#之后的内容均是注释。
echo 'Hello World'
多行注释
格式:
:<<EOF
第一行注释
第二行注释
第三行注释
EOF
其中EOF可以换成其它任意字符串。例如&#xff1a;
:<<abc
第一行注释
第二行注释
第三行注释
abc:<<!
第一行注释
第二行注释
第三行注释
!
3.3 变量
定义变量
定义变量&#xff0c;不需要加$符号&#xff0c;例如&#xff1a;
name1&#61;&#39;yxc&#39;
name2&#61;"yxc"
name3&#61;yxc
使用变量
使用变量&#xff0c;需要加上符号&#xff0c;或者符号&#xff0c;或者符号&#xff0c;或者{}符号。花括号是可选的&#xff0c;主要为了帮助解释器识别变量边界。
name&#61;yxc
echo $name
echo ${name}
echo ${name}acwing
只读变量
使用readonly或者declare可以将变量变为只读。
name&#61;yxc
readonly name
declare -r name name&#61;abc
删除变量
unset可以删除变量。
name&#61;yxc
unset name
echo $name
变量类型
-
自定义变量&#xff08;局部变量&#xff09;
子进程不能访问的变量
-
环境变量&#xff08;全局变量&#xff09;
子进程可以访问的变量
自定义变量改成环境变量&#xff1a;
acs&#64;9e0ebfcd82d7:~$ name&#61;yxc
acs&#64;9e0ebfcd82d7:~$ export name
acs&#64;9e0ebfcd82d7:~$ declare -x name
环境变量改为自定义变量&#xff1a;
acs&#64;9e0ebfcd82d7:~$ export name&#61;yxc
acs&#64;9e0ebfcd82d7:~$ declare &#43;x name
字符串
字符串可以用单引号&#xff0c;也可以用双引号&#xff0c;也可以不用引号。
单引号与双引号的区别&#xff1a;
单引号中的内容会原样输出&#xff0c;不会执行、不会取变量&#xff1b;
双引号中的内容可以执行、可以取变量&#xff1b;
name&#61;yxc
echo &#39;hello, $name \"hh\"&#39;
echo "hello, $name \"hh\""
获取字符串长度
name&#61;"yxc"
echo ${#name} # 输出3
提取子串
name&#61;"hello, yxc"
echo ${name:0:5}
3.4 默认变量
文件参数变量
在执行shell脚本时&#xff0c;可以向脚本传递参数。$1是第一个参数&#xff0c;$2是第二个参数&#xff0c;以此类推。特殊的&#xff0c;$0是文件名&#xff08;包含路径&#xff09;。例如&#xff1a;
创建文件test.sh&#xff1a;
#! /bin/bashecho "文件名&#xff1a;"$0
echo "第一个参数&#xff1a;"$1
echo "第二个参数&#xff1a;"$2
echo "第三个参数&#xff1a;"$3
echo "第四个参数&#xff1a;"$4
然后执行该脚本&#xff1a;
acs&#64;9e0ebfcd82d7:~$ chmod &#43;x test.sh
acs&#64;9e0ebfcd82d7:~$ ./test.sh 1 2 3 4
文件名&#xff1a;./test.sh
第一个参数&#xff1a;1
第二个参数&#xff1a;2
第三个参数&#xff1a;3
第四个参数&#xff1a;4
其它参数相关变量
参数 说明
$# 代表文件传入的参数个数&#xff0c;如上例中值为4
$* 由所有参数构成的用空格隔开的字符串&#xff0c;如上例中值为"$1 $2 $3 $4"
$&#64; 每个参数分别用双引号括起来的字符串&#xff0c;如上例中值为"$1" "$2" "$3" "$4"
$$ 脚本当前运行的进程ID
$? 上一条命令的退出状态&#xff08;注意不是stdout&#xff0c;而是exit code&#xff09;。0表示正常退出&#xff0c;其他值表示错误
$(command) 返回command这条命令的stdout&#xff08;可嵌套&#xff09;
&#96;command&#96; 返回command这条命令的stdout&#xff08;不可嵌套&#xff09;
3.5 数组
数组中可以存放多个不同类型的值&#xff0c;只支持一维数组&#xff0c;初始化时不需要指明数组大小。
数组下标从0开始。
定义
数组用小括号表示&#xff0c;元素之间用空格隔开。例如&#xff1a;
array&#61;(1 abc "def" yxc)
也可以直接定义数组中某个元素的值&#xff1a;
array[0]&#61;1
array[1]&#61;abc
array[2]&#61;"def"
array[3]&#61;yxc
读取数组中某个元素的值
格式&#xff1a;
${array[index]}
例如&#xff1a;
array&#61;(1 abc "def" yxc)
echo ${array[0]}
echo ${array[1]}
echo ${array[2]}
echo ${array[3]}
读取整个数组
格式&#xff1a;
${array[&#64;]}
${array[*]}
例如&#xff1a;
array&#61;(1 abc "def" yxc)echo ${array[&#64;]}
echo ${array[*]}
数组长度
类似于字符串
${#array[&#64;]}
${#array[*]}
例如&#xff1a;
array&#61;(1 abc "def" yxc)echo ${#array[&#64;]}
echo ${#array[*]}
3.6 expr命令
expr命令用于求表达式的值&#xff0c;格式为&#xff1a;
expr 表达式
表达式说明&#xff1a;
- 用空格隔开每一项
- 用反斜杠放在shell特定的字符前面&#xff08;发现表达式运行错误时&#xff0c;可以试试转义&#xff09;(需要转义的&#xff1a;* 、() 、 > 、 <&#61; 、| 、&)
- 对包含空格和其他特殊字符的字符串要用引号括起来
- expr会在stdout中输出结果。如果为逻辑关系表达式&#xff0c;则结果为真&#xff0c;stdout为1&#xff0c;否则为0。
- expr的exit code&#xff1a;如果为逻辑关系表达式&#xff0c;则结果为真&#xff0c;exit code为0&#xff0c;否则为1。
字符串表达式
- length STRING
- 返回STRING的长度
- index STRING CHARSET
CHARSET中任意单个字符在STRING中最前面的字符位置&#xff0c;下标从1开始。如果在STRING中完全不存在CHARSET中的字符&#xff0c;则返回0。 - substr STRING POSITION LENGTH
返回STRING字符串中从POSITION开始&#xff0c;长度最大为LENGTH的子串。如果POSITION或LENGTH为负数&#xff0c;0或非数值&#xff0c;则返回空字符串。
示例&#xff1a;
str&#61;"Hello World!"echo &#96;expr length "$str"&#96;
echo &#96;expr index "$str" aWd&#96;
echo &#96;expr substr "$str" 2 3&#96;
整数表达式
expr支持普通的算术操作&#xff0c;算术表达式优先级低于字符串表达式&#xff0c;高于逻辑关系表达式。
- -&#43;
加减运算。两端参数会转换为整数&#xff0c;如果转换失败则报错。
- / %
乘&#xff0c;除&#xff0c;取模运算。两端参数会转换为整数&#xff0c;如果转换失败则报错。 - () 可以该表优先级&#xff0c;但需要用反斜杠转义
示例&#xff1a;
a&#61;3
b&#61;4echo &#96;expr $a &#43; $b&#96;
echo &#96;expr $a - $b&#96;
echo &#96;expr $a \* $b&#96;
echo &#96;expr $a / $b&#96;
echo &#96;expr $a % $b&#96;
echo &#96;expr \( $a &#43; 1 \) \* \( $b &#43; 1 \)&#96;
逻辑关系表达式
- |
如果第一个参数非空且非0&#xff0c;则返回第一个参数的值&#xff0c;否则返回第二个参数的值&#xff0c;但要求第二个参数的值也是非空或非0&#xff0c;否则返回0。如果第一个参数是非空或非0时&#xff0c;不会计算第二个参数。 - &
如果两个参数都非空且非0&#xff0c;则返回第一个参数&#xff0c;否则返回0。如果第一个参为0或为空&#xff0c;则不会计算第二个参数。 - <<&#61; &#61; &#61;&#61; !&#61; >&#61; >
比较两端的参数&#xff0c;如果为true&#xff0c;则返回1&#xff0c;否则返回0。”&#61;&#61;”是”&#61;”的同义词。”expr”首先尝试将两端参数转换为整数&#xff0c;并做算术比较&#xff0c;如果转换失败&#xff0c;则按字符集排序规则做字符比较。 - () 可以该表优先级&#xff0c;但需要用反斜杠转义
示例&#xff1a;
a&#61;3
b&#61;4echo &#96;expr $a \> $b&#96;
echo &#96;expr $a &#39;<&#39; $b&#96;
echo &#96;expr $a &#39;>&#61;&#39; $b&#96;
echo &#96;expr $a \<\&#61; $b&#96; c&#61;0
d&#61;5echo &#96;expr $c \& $d&#96;
echo &#96;expr $a \& $b&#96;
echo &#96;expr $c \| $d&#96;
echo &#96;expr $a \| $b&#96;
3.7 read命令
read命令用于从标准输入中读取单行数据。当读到文件结束符时&#xff0c;exit code为1&#xff0c;否则为0。
参数说明
- -p: 后面可以接提示信息
- -t&#xff1a;后面跟秒数&#xff0c;定义输入字符的等待时间&#xff0c;超过等待时间后会自动忽略此命令
实例&#xff1a;
acs&#64;9e0ebfcd82d7:~$ read name
acwing yxc
acs&#64;9e0ebfcd82d7:~$ echo $name
acwing yxc
acs&#64;9e0ebfcd82d7:~$ read -p "Please input your name: " -t 30 name
Please input your name: acwing yxc
acs&#64;9e0ebfcd82d7:~$ echo $name
acwing yxc
3.8 echo命令
echo用于输出字符串。命令格式&#xff1a;
echo STRING
显示普通字符串
echo "Hello AC Terminal"
echo Hello AC Terminal
显示转义字符
echo "\"Hello AC Terminal\""
echo \"Hello AC Terminal\"
显示变量
name&#61;yxc
echo "My name is $name"
显示换行
echo -e "Hi\n"
echo "acwing"
输出结果&#xff1a;
Hiacwing
显示不换行
echo -e "Hi \c"
echo "acwing"
输出结果&#xff1a;
Hi acwing
显示结果定向至文件
echo "Hello World" > output.txt
原样输出字符串&#xff0c;不进行转义或取变量(用单引号)
name&#61;acwing
echo &#39;$name\"&#39;
输出结果
$name\"
显示命令的执行结果
echo &#96;date&#96;
输出结果&#xff1a;
Wed Sep 1 11:45:33 CST 2021
Linux基础课