Shell 是一门脚本语言(又称解释型语言),Shell 其实就是一个纯文本文件,通常以【#!/bin/bash】开始。脚本自上而下,从左至右分析并执行,其中【#】后面的为注释。脚本有以下几种运行方式:(1) 【bash shell.sh】 (2)【sh shell.sh】(3)【./shell.sh】当前目录下执行,用户必须有可执行权限 (4)【. ./shell.sh】或【source shell.sh】无执行权限也能执行,【source】一般用于读取配置文件使其立即生效。
第一个 Shell 脚本:
#!/bin/bash # show Hello World # 2016-06-06 cyhe echo "Hello World" exit 0
1. 变量
在Shell中,变量是弱类型的(存放各类数据,比如文件名,路径名等),而且无论值是否加引号,默认都是字符串。变量名只能以字母或下划线开头,并且区分大小写,长度无限制。变量分为局部变量和环境变量。当登陆 Linux取得一个交互Shell后,我们可以在这个 Shell(父Shell) 中管理系统、创建 子Shell、定义变量。子Shell 能够继承 父Shell 的环境变量,不会继承自定义变量,在Shell 可以使用【export】将自定义变量上升为环境变量,供子程序使用。
(1)局部变量
局部变量的作用域在 Shell 内部,其他shell不能访问,随脚本结束而消亡。可以使用 local 可以显式声明局部变量,常用于函数内部。
(2)环境变量
环境变量又称为全局变量,Shell 中变量默认就是全局的。为了有一个较好的执行环境,在编写 Shell 之前可以先设置好一些重要的环境变量,比如PATH,LANG,可以方便的下达一些命令,而不必使用绝对路径。比如,在使用 crontab 定期执行脚本自动备份Oracle数据库(或其他)时,Oracle相关的环境变量(ORACLE_HOME, NLS_LANG…)一般在oracle用户下,此时脚本不使用 export 声明Oracle相关变量,将不能执行。
(3)变量的赋值与取值
#!/bin/bash
# 变量名=值(var=value 等号两边没有空格),使用 【$var/${var}】 符号取值
# 如果有空格就必须使用引号
name=cyhe
name="cy he"
echo $name
echo ${name}
Shell 中预定义的特殊变量和特殊字符:
| 变量 | 说明 |
| $0 | 脚本名称 |
| $n | 脚本或函数参数,$1 第一个参数,$2 第二个以此类推 |
| $? | 脚本或命令返回值 |
| $*,$@ | 显示所有参数 |
| 通配符 | *:任意长度字符串,不包括点号和斜线 ?:匹配单一字符 []:匹配其中的任一个 |
| ‘’ | 单引号:强引用,作为普通字符处理 |
| “” | 双引号:解析字符串里的【$,,`】特殊用途 |
| # | 注释 |
| $(命令) | 可以把命令的输出赋值给变量,支持嵌套 |
| ; | 分号,执行多个命令,# date; ls -l |
2. 算术运算
Shell 只支持整数计算
| 命令 | 用法 |
| let | #!/bin/bash a=1 b=2 let c=a+b 自增 let a++ 自减 let a-- |
| [ ] | c=$[a+b] 或 c=$[ $a+2 ] 取模 $[5%2] 幂运算 $[2**3] |
| (()) | c=$(( a+b )),c=$(( (5+2)*3 )),(( a++ )) 常用 |
| bc | 高精度计算语言,支持比较运算和逻辑运算 |
3. 数组
在 shell 中,数组中元素类型可以不同,数组的长度也不限制,可以随时随地增加数组元素,但是只支持一维数组,使用整数作索引,也可以使用字符串作索引。
#!/bin/bash
# 数组相关操作
# author: cyhe 2016-11-08
nums=(1 2 3 4 5) # 定义数组
weeks=("sun" "mon" "thu")
weeks[3]="wed" # 随时随地增加数据
weeks[4]="thur"
echo ${#weeks[@]} # 数组长度 or ${#weeks[*]}
echo ${#weeks[4]} # 数组元素的长度
echo ${weeks[@]} # 数组所有元素
echo ${weeks[0]} # 取第 0 个元素
unset weeks[1] # 取消第 1 个元素
4. 判断
在编程时,会经常根据字符串是否为空,文件是否存在诸如此类的问题,来控制程序的流程,在 Shell 中可以使用【test expression】命令直接测试。此外可以利用判断符号【[ ](中括号)】进行数据的判断,在使用中括号有几个关键的地方:括号内的每个元素都要用空格分隔;括号内的变量,最好都用双引号括起来;括号内容的常量,最好都用单或双引号括起来。常与 if 语句搭配使用。
常用测试判断条件
| 标志 | 说明 |
| 1. 文件类型 | [ –e filename ] |
| -e | 文件是否存在 |
| -f | 文件是否存在且为文件(file) |
| -d | 文件是否存在且为目录(directory) |
| 2. 文件权限 | [ –r filename ] |
| -r/w/x | 文件是否存在且有相应的权限 |
| -s | 文件是否存在且为非空白文件 |
| 3. 比较文件 | [ file1 -nt file2 ] |
| -nt/-ot | 判断file1是否比file2新(new than)/旧(older than) |
| -ef | 是否为同一文件 |
| 4. 比较整数 | [ n1 -eq n2 ] |
| -eq/-ne | 相等(equal)/不相等(not equal) |
| -gt/-lt | 大于(greater than)/小于(less than) |
| -ge/-le | 大于等于/小于等于 |
| 5. 判断字符串 | |
| [ –z string ] | 是否为空 |
| [ –n string ] | 是否不空 |
| [ str1 == str2 ] | 是否相等 |
| [ str1 != str2 ] | 是否不等 |
| 6. 逻辑运算 | |
| -a | and,[ –r file –a –x file ],同时有rx权限 |
| -o | or,[ –r file –o –x file ],有r或x权限 |
| ! | 非,[ ! –x file ],不具有x权限 |
| && | 与,[ –r file ] && [ –x file ] |
| || | 或,[ –r file ] || [ –x file ] |
| ! | 非,! [ –e file ] |
【if/else判断】
#!/bin/bash
str="Hello World"
file=/opt/shell.log
# 单层条件判断 这里分号的作用是连续执行命令,不用换行了
if [ -z "$str" ]; then # 若字符串为空
# TODO
fi
# 一个分支
if [ -f "$file" ]; then # 文件是否存在
# TODO
else
# TODO
fi
# 多个分支
if [ -z "$1" ]; then # 若第一个参数为空
# TODO
elif [ "$1" == "" ]; then # 是否为空串
# TODO
else
# TODO
fi
【case … esac判断】
#!/bin/bash
case $1 in
"start") # 第一个变量内容,关键字右小括号
echo "start"
;; # 该程序段结束
"stop")
echo "start"
;;
"restart")
echo "stop"
echo "start"
;;
*) # 默认匹配
echo "Usage: $0 {start|stop|restart}"
exit 1
;;
esac # 结束
5. 循环
【for 循环】
#!/bin/bash
# 1. 用来遍历
nums=(1 2 3 4 5);sum=0
for n in ${nums[@]} # 遍历数组求和
do
sum=$(($sum + $n))
done
# 遍历 1-5 的数字
for n in {1..5} # 或1 2 3 4 5 或$(seq 1 5)
do
echo $n
done
# 2. 带条件的for循环,求1-100的和
sum=0
for (( i=0; i<=100; i++ ))
do
sum=$(( $sum+$i )) # 或 let sum=sum+i
done
# 死循环
for ((;1;))
【while 循环】
#!/bin/bash
while [ condition ]
do
# TODO
done
# while一个经典用法,按行读取文件
cat foo.txt | while read line # 运行时产生 3 个shell:cat,管道,while
do
echo $line
done
# 死循环
while true
while 1
6. 函数
函数就是自定义的一系列执行命令,一般设置返回值(0正确,非0错误),最主要还是用来代码复用。
#!/bin/bash
# 1. 函数定义与返回值
function fun_name(){ # function 关键字可省略
# TODO
echo $@ # 打印所有参数 无输出 0
return 0
}
fun_name # 函数调用,只调用函数名即可
echo $? # 取函数返回值,可赋值给变量
# 2. 参数
# fun_name arg1 arg2 ...
fun_name a b c
# 3. 变量
# shell中变量默认是全局的,可使用 local 定义局部变量
n=5
fun(){
local n=1
let n++
}
fun # 调用函数不影响 n 的值
echo $n
# $0 代表函数名 $1 第一个参数 $2 第二个 一次类推
# 这里的参数与脚本本身的参数 毫无关系
7. 调试脚本debug
sh –n shell.sh 检查语法问题,不执行脚本
sh –v shell.sh 执行前先输出脚本内容到屏幕
sh –x shell.sh 将脚本执行过程输出到屏幕