Shell 是一种特殊功能的程序,它介于用户和 unix/linux 操作系统内核程序之间的一个接口,通过 SSH 服务连接到 shell 就可进行远程操作了。
shell 中的变量
- 用户自定义变量, 用户在 bash 窗口或者 shell 脚本中临时定义的变量
- 位置变量, $0,$1,$2….用来获取传入参数及程序名称($0)
- 预定义变量(系统变量), $#,$*,$@,$?,$$,$!这些变量只能使用不能修改
- 环境变量, 系统中默认存在的变量如:$PWD, $PATH, 可以通过export导出, 导出的变量可以穿透多层 bash/shell。一般变量可以直接在 shell 中赋值创建(如:a=1)或使用set a=1,二者是等价的,如果想删除变量使用unset <变量名>。
| 变量名 | 含义 | 
|---|---|
| $# | 命令行参数数量 | 
| $0 | 当前程序(脚本)名称, 带路径, 通过 sh/bash 执行的脚本名称中不含路径 | 
| $? | 前一个命令或函数的返回值 | 
| $* | 以 “参 1 参 2 参 3 …” 返回所有参数, 此时所有参数是以空格分割的一个字符串 | 
| $@ | 以"参 1" “参 2” “参 3” …返回所有参数, 此时每个参数都是一个字符串 | 
| $$ | 本程序(脚本)的进程号 PID) | 
| $! | 上个命令的进程号 | 
几个特殊引用
- 双引号: 括起来的字符除了 $(美元符)、\(反引号)、"(双引号)、`(反引号)之外其他的均视为普通字符对待.
- 单引号: 括起来的字符都视为普通字符,普通字符直接被打印
- 反引号: 反引号括起来的字符串会被 shell 解释为命令, 就像在 shell 中输入命令一样, 注意: 在反双引号中不能用命令的别名
脚本中常用变量替换
- ${var}取出 var 变量的值
- ${var:-var2}var 为空或已被删除或未定义, 则表达式返回 var2, 不改变 var 的值
- ${var:+var2}var 未定义, 则返回 var2, 不改变 var 的值
- ${var:=var2}var 为空或已被删除或未定义, 则表达式返回 var2, 改变 var 的值!, 如果已被删除或未定义会重新创建变量
- ${var:?msg}如果 var 为空或已被删除或未定义, 则在控制台标准输出打印 msg, 可以用来检测变量 var 是否被赋值. 当出现这种替换, 脚本将停止运行.
字符串变量的截取
- ${#str}求 str 的串长
- ${str:3}从第 3 个字符截取到最后一个字符(含第 3 个)
- ${str:3:5}从第 3 个字符截取到第 5 个字符(含第 3 第 5 个)
sehll 中变量计算
- $(()), 在两个括号中可以放置需要计算的变量表达式
- $[], 在中括号中放入表达式
- let, let sum=1+2
- expr, 如: sum=- "expr 1 + 2", 注意:- +好两端要有空格!
read – 读取变量命令
该用在 shell 脚本中, 读取一个变量并赋值, 如果指定多个变量但输入参数数量超过时, 多的参数赋值在最后个接收的变量. 参数:
- -p 后面跟提示信息, 当用户输入时显示提示信息
- -n 指定 read 读入文本长度
- -s 静默输入, 输入字符时不显示在屏幕上
echo
常用参数:
- -n 在屏幕打印后不换行
- -e 启用反斜杠转义功能,对\n之类转义字符进行转义,否则作为普通字符输出
- -E 此项为缺省项, 不对转义字符做特殊处理
常用转义字符
| 转义字符 | 含义 | 
|---|---|
| \n | 换行 | 
| -r | 回车, 换到下一行并将光标置于行首 | 
| \r | 制表符 | 
| \b | 退格,左删除 | 
| \\ | 斜杠本身 | 
shell 中比较和判断
在 shell 中 0 代表的是真, 非零代表为假, 这一点与 C/C++截然相反的.
数值比较表达式
test 表达式的两种形式: test var -eq var2 [ var -eq var2 ]
| test 表达式 | 含义 | 
|---|---|
| var -eq var2 | 判断两变量是否相等 | 
| var -ge var2 | 判断 var 是否大于等于 var2 | 
| var -le var2 | 判断 var 是否小于等于 var2 | 
| var -gt var2 | 判断 var 是否大于 var2 | 
| var -lt var2 | 判断 var 是否小于 var2 | 
| var -ne var2 | 判断 var 是否不等于 var2 | 
类型判断表达式
| test 表达式 | 含义 | 
|---|---|
| -d file | 判断 file 是否为目录 | 
| -f file | 判断 file 是否为文件 | 
| -r file | 判断文件是否可读 | 
| -w file | 判断文件是否可写 | 
| -x file | 判断文件是否可以执行 | 
| -s file | 判断文件内容长度是否大于 0 | 
判断字符串
| test 表达式 | 含义 | 
|---|---|
| = | 等于则为真 | 
| != | 不相等则为真 | 
| -z 字符串 | 字符串的长度为零则为真 | 
| -n 字符串 | 字符串的长度不为零则为真 | 
逻辑判断表达式
| test 表达式 | 含义 | 
|---|---|
| !var | var=假/0, 返回真 | 
| var1 -a var2 | var1 和 var2 同时为真才返回真,否则返回假 | 
| var1 -o var2 | var1 和 var2 至少有一个为真是返回真,否则返回假 | 
注意事项 变量测试表达式一般不单独使用, 常常作为 for/if 的条件.
shell 的分支循环结构
if 结构
#!/bin/bash
let var=2
# 常用写法
if [ $var -eq 0 ]
then
    echo "var is 0"
fi
# 常用写法2
if [ $var -eq 0 ]; then
    echo "var is 0"
fi
# 直接写test方式
if test $var -eq 0
then
    echo "var is 0"
fi
# if-else
if [ $var -eq 0 ]
then
   echo "var is 0"
else
   echo "var is other number."
fi
# if-elif-else
if [ $var -eq 0 ]
then
   echo "var is 0"
elif [ $var -eq 1 ]
then
   echo "var is 1"
else
   echo "var is other number."
fi
case 结构
case 结构与常见语言的 switch 语句相似, 不同的是每个 case 分支结束时用双分号;;.
#!/bin/bash
read -p "please intput a animal name: " a
# 匹配字符串
case $a in
cat)
    echo "this is a cat."
    ;;
fish)
    echo "this is a fish."
    ;;
*)
    echo "another animal."
esac
# 匹配数字
case $a in
1)
    echo "match number 1"
    ;;
2)
    echo "match number 2"
    ;;
*)
    echo "unknow number: $a"
esac
for 结构
#!/bin/bash
echo "-----------"
echo "all args:$*"
echo "all args:$@"
echo "-----------"
# 常用, $*可以替换为$@,都获取执行脚本时所有参数
for a in $*
do
    echo "$a"
done
# 循环1到5, in后面的{}中为一组内容,数字或文本都可
# {1..5} 等价于 {1,2,3,4,5}
for a in {1,2,3,4,5}
do
    echo "$a"
done
# 循环1到5
for a in 1 2 3 4 5
do
    echo "$a"
done
# C语言风格, 循环1~5
for (( i=1; i<=5; i++ ))
do
    echo "$i"
done
# -------------
# 输出当前目录中以.sh结尾的文件
for n in `ls *.sh`
{
    echo $n
}
# 输出 1~10,{1..10} 产生1~10序列,等价于 seq 10
for a in {1..10}
{
    echo $a
}
while 结构
在 while 改变变量的值使用let 或 var=$[express], 其中 var 为一个变量,express 为一个运算表达式,如 1+2
**#!/bin/bash
let var=1
# 常用, 循环1~5
while [ $var -le 5 ]
do
    echo "$var"
    let var=$var+1
done
let var=1
while [ $var -le 5 ]
do
    echo "$var"
    let var++
done
until
当条件为假时执行循环, 其使用同while.
循环控制语句
中断语句:
- break: 用来终止循环, 如果是前嵌套循环 break 后可以跟一个数字 n, 表示退出第 n 重循环(最内的循环为第一重)
- continue: 用来跳过(忽略)本次循环, 如果是嵌套循环, continue 后可以跟一个数字 n, 表示回到第 n 重循环的顶部
#!/bin/bash
let var=1
# 常用, 循环0~5
while [ $var -le 5 ]
do
    if [ $var -eq 2 ]
    then
        echo "now var=$var, break while."
        break
    fi
    echo "$var"
    let var=$var+1
done
echo "----------------------"
let var=1
while [ $var -le 5 ]
do
    if [ $var -eq 2 ]
    then
        echo "now var=$var, conticned the second cycle."
        break
    fi
    echo "$var"
    let var++
done
函数
函数的在 shell 中是独立的一段代码,实现一些功能的复用。
函数要点:
- 函数可以接收参数,同 shell 脚本接收参数方式相同,使用$*,$@,$1
- 函数体中获取参数的变量($*,$@,$1)获取的是函数自己的参数
定义方式
#!/bin/bash
# 定义函数方式1
fun1()
{
    echo "called fun1"
}
# 定义函数方式2
function fun2
{
    echo "called fun2"
}
#调用函数
fun1
fun2
函数实现递归
#!/bin/bash
# 通过读取脚本的第一输入参数值求其阶乘, 并输出结果
# 递归求阶乘函数
function jie
{
    if [ $1 -le 1 ]
    then
        echo 1
    else
        x=`jie $[$1-1]`
        echo $[$1*x]
    fi
}
# 调用计算
res=$(jie $1) # 此处函数jie接收的第一个参数为此脚本运行时的第一个参数
echo "res=$res" # 接收echo输出的返回值
数组
在 shell 中数组与高级编程语言中的数组时有区别的, 在声明方面可以用declare -a arr声明 arr 数组, 参数-a 是声明数组的意思, 除此之外 declare 还可以声明函数, 其次当我们使用 declare 声明一个数组时,下标越界不会有报错提示.
declare 后的参数
-a 声明数组 -i 声明 integer 整数 -r 声明变量定义为只读 -f 定义为函数
数组的使用
#!/bin/bash
declare -a arr[10]
# 即使下标越界(i>=10)也不会报错
for ((i=0;i<10;i++))
{
    arr[i]=$[i+1]
}
# 输出
for ((k=0;k<10;k++))
{
    echo ${arr[k]}
}
总结
学完以上只是对 shell 编程的一个入门, 如果把一下常用的命令(如: awk、sed、grep、find)结合进脚本,功能就会大大增强.
在写 test 判断表达式时尽量每个部分之间都一个空格,以免出现格式操作。注意: expr 的运算符两边是必须要留空格的。总之多留一些空格可以避免一些语法错误。
for 可以遍历的内容很多, 不仅仅是数组, 在 in 后可以放置命令的执行结果(以空格分割)