注:阅读本文请自带基本编程技能(至少会C)
Hello world
shell脚本和一般程序并没有太多的差别,脚本文件基本形态,以#开头表示注释,直到行末,先来个Hello world吧
#!/bin/sh # 声明shell脚本解释器,没有声明则使用默认的shell解释器,不同的shell解释器语法有一定的区别,所以都要加上此句
a = "hello world" # 变量不需要定义,无数据类型
echo $a # 输出 hello world
echo "output ${a}abc" # 输出 output hello worldabc (如果变量后面还有字符则用{}括起来,$表示取变量的值)
保存为test文件,使用命令行执行./test,输出为output hello worldabc
系统变量
变量 | 描述 |
---|---|
$HOME | 当前用户登录的子目录路径 |
$PATH | 以冒号分隔,用来搜索命令的目录清单 |
$PS1 | 命令行提示符,通常是'$'字符 |
$PS2 | 辅助提示符,用来提示后续的输入,通常是'>' |
$IFS | 输入区的分隔符,通常是空格、制表符和换行符 |
$0、$1、$2... | 脚本程序的参数,$0为shell脚本的名字,后续为脚本程序的输入参数 |
$* | 全体参数组成的清单,用$IFS的第一个字符(空格)隔开 |
$@ | $*的一种变量,不使用$IFS变量 |
$# | 传递到脚本程序或子程序的参数个数 |
$$ | 该shell程序的进程id,一般用来创建临时文件,如tempfile_$$ |
条件表达式
类型 | 表达式 | 描述 |
---|---|---|
字符串比较 | "字符串1" = "字符串2" "字符串1" != "字符串2" -n "字符串" -z "字符串" |
相等(值) 不等 非空 为空 |
算数比较 | 表达式1 -eq 表达式2 表达式1 -ne 表达式2 表达式1 -gt 表达式2 表达式1 -ge 表达式2 表达式1 -lt 表达式2 表达式1 -le 表达式2 !表达式 |
相等(值) 不等 大于 大于或等于 小于 小于或等于 取反 |
文件条件测试 | -d 文件名(路径) -e 文件名 -f 文件名 -r 文件名 -s 文件名 -w 文件名 -x 文件名 -g 文件名 -u 文件名 |
是一个目录 文件存在 是普通文件 文件可读 文件长度不为0 文件可写 文件可执行 文件的set -group-id位被设置 文件的set -userid被设置 |
语法规则
if语句
if [表达式] # 表达试多个组合情况下和C一样,前面已经成立的后面就不再执行
then
[语句组]
elif [表达式] # else if
then
[语句组]
else
[语句组]
fi
for语句
for 变量 in 值的范围(可用通配符,命令的输出等)
do
[语句组]
done
例:
a=2
for a in 1 2 3 "ab"c # 循环体执行4次
do
echo $a
done
输出为:123abc
while语句
while [表达式]
do
[语句组]
done
until语句
until [表达式] # 表达式为真时退出循环
do
[语句组]
done
case语句
case 变量 in
情况1[|情况2...]) 语句组 ;; # ;;表示break
情况3[|情况4...]) 语句组 ;;
...
*) 语句组 ;; # 相当于default
esac
例:
#!/bin/sh
read a # 等待输入
case "$a" in
y|yes) echo "yes" ;;
n|no) echo "no" ;;
*) echo "default"
esac
函数
定义:函数名() (没有返回值类型、参数,写在调用前面)
#!/bin/sh
fun(){
echo $1
echo $2
return 0
}
echo "call fun"
a = fun aa bb
echo $a
理论输出(获取返回值可能出错):call fun aa bb 0
获取返回值后函数中的echo可能不会显示,全被捕获到返回值中(待验证)
函数中声明局部变量在变量前加local
fun(){
local a
a = 10
...
}
break命令
用于跳出循环(for while until),可以带一个参数,表示跳出循环的层数
for [表达式]
do
[语句组]
if [表达式]
then
beeak; # ;可有可无
fi
done
:命令
空操作,相当于true或汇编中的nop(),但效率高,用于while : (也就是我们C中常用的while(true))
continue命令
用法和C相同(结束本次循环继续下一个)
.命令
代表当前目录,例如执行当前目录下的test文件
./test
eval命令
将表达式得到的结果以shell的形式再执行一次
#!/bin/sh
foo = 10
x = foo
y = '$'$x
echo $y
输出:$foo
#!/bin/sh
foo = 10
x = foo
eval y = '$'$x # 等效于y = $(($x)) (应该是y = eval '$'$x吧,待验证)
echo $y
输出:10
exec命令
执行另一个shell程序,执行完不会再返回,exec命令后面的语句不会被执行
#!/bin/sh
exec echo "execute"
echo "next" # 不会被执行
exit命令
退出程序,可带一个参数,0表示正常退出
export命令
导出,相当于C中.h文件的变量声明,给不同的文件使用
export1文件
#!/bin/sh
foo = "aaa"
export foo
./export2
export2文件
#!/bin/sh
echo "$foo"
输出:aaa
set命令
设置参数到s1、s2、s3...
#!/bin/sh
echo $(data) # data获取系统时间
set $(data)
echo $2 # $2中保存着月份
unset命令
删除变量
#!/bin/sh
foo = "aaa"
unset foo
echo $foo # 输出为空