1.Shell脚本
1.1 格式
首行 #!/bin/bash
指定解释器
1.2 注释
'#'开头的行,'#!'是例外
此外,# 是特殊字符,可以出现在一些参数代换结构和在数值常量表达式中,具有特殊含义,而不会开启一个注释。‘#’也不会开启一个注释。
1.3 函数
function funname(){…}
或者
funname()
{
statements;
}
只需要使用函数名就可以调用某个函数:funname
参数可以传递给函数,使用方法就好像函数是个新脚本一样:
funname arg1 arg2...; #传递参数
在函数中使用传入的参数:$1 第一个参数;$@ 所有参数。
其中:"$@"被扩展成"$1""$2""$3";
"$*"被扩展成"$1c$2c$3",即一个字符串。c为IFS的第一个字符。
有时我们需要知道命令或者函数的执行状态,用$?可以查看前一个命令的返回值,如果命令成功退出,那么退出状态为0,否则非0。
1.4 正文部分
流程控制+命令
1.5 执行:修改权限
转为可执行程序
chmod +x ./test.sh #使脚本具有执行权限
./test.sh #执行脚本
1.6 流程控制
条件语句
if :
if condition
then
command1
command2
...
commandN
fi
if `ps -ef | grep ssh`;
then
echo hello;
fi
if else-if else :
if condition1
then
command1
elif condition2
command2
else
commandN
fi
循环语句
for :
for var in item1 item2 ... itemN
do
command1
command2
...
commandN
done
for var in item1 item2 ... itemN; do command1; command2… done;
while :
i=1;total=0;
while [ $i -le 10 ]
do
let total+=i
let i++
echo $total,$i
done
i=1; total=0;
while((i<=10))
do
((total+=i, i++))
echo $total,$i
done
case语句: case语句可以用户处理自定义参数。
case $num in
1) echo "January";; #双分号结束
2) echo "Feburary";;
5) echo "may" #每个case可以有多条命令
echo "sdfd"
echo "sdf";; #但最后一条命令一定是双分号结束
*) echo "not correct input";; #*)是其他值、default的意思
esac
1.7 while read line
while read line; do something ; done
1.8 参数处理
a) "$*"将所有的参数解释成一个字符串,而"$@"是一个参数数组。
b) Shell内建函数getopts “:a:bc” opt
主要变量:
$OPTIND : 存储所处理的选项在参数列表中的位置
$OPTARG : 存储相应选项所带的参数
例子:
while getopts ":a:b:cef" opt
do
case $opt in
a)echo "the $OPTIND has arg:$OPTARG";;#$OPTIND=3
b)echo "the b has arg:$OPTARG";;
c | e | f)echo "the $opt has no arg";;
\?)echo "the $opt is invalid param";;
esac
done
c) shift n 将位置命令左移n个
1.9 条件判断
条件判断应该放进方括号里,且方括号两边都应该留有空格。 [ ]
a) 字符串判断
字符串比较时,最好用双中括号,因为有时候采用单中括号会产生错误,所以最好避开它们。[[ $str1 = $str2 ]]
= 当两个串有相同内容、长度时为真
!= 当串str1和str2不等时为真
-n 当串的长度大于0时为真(串非空)
-z 当串的长度为0时为真(空串)
b) 数值判断
-eq 两数相等为真
-ne 两数不等为真
-gt int1大于int2为真
-ge int1大于等于int2为真
-lt int1小于int2为真
-le int1小于等于int2为真
c) 文件判断
-e file 若文件存在,则为真
-d file 若文件存在且是一个目录,则为真
-b file 若文件存在且是一个块特殊文件,则为真
-c file 若文件存在且是一个字符特殊文件,则为真
-f file 若文件存在且是一个规则文件,则为真
-g file 若文件存在且设置了SGID位的值,则为真
-h file 若文件存在且为一个符合链接,则为真
-k file 若文件存在且设置了"sticky"位的值
-p file 若文件存在且为一已命名管道,则为真
-r file 若文件存在且可读,则为真
-s file 若文件存在且其大小大于零,则为真
-u file 若文件存在且设置了SUID位,则为真
-w file 若文件存在且可写,则为真
-x file 若文件存在且可执行,则为真
-o file 若文件存在且被有效用户ID所拥有,则为真
d) 逻辑判断
! 非
-a 与 &&
-o 或 ||
if [ $v –ne 0 –a $v –lt 2 ] 等价 if [ $v –ne 0 ] && [ $v –lt 2 ]
if [ $v –ne 0 –o $v –lt 2 ] 等价 if [ $v –ne 0 ] || [ $v –lt 2 ]
条件判断部分可能会变得很长,一个优化的小技巧是利用&&和||运算符。
if condition
then
command1
else
command2
fi
[ condition ] && command1 || command2
这样就用一行代替了上面的5行而实现的功能完全相同。
如果命令有多个,可以用{}括起来,当做一个命令块。
这样可以使判断语句变得非常简洁。
1.10 &&、||
cmd1 && cmd2
表示,当cmd1执行成功后,就执行cmd2,否则不执行。
cmd1 || cmd2
表示,当cmd1执行失败后,就执行cmd2,否则不执行。
2.变量
2.1 系统变量
$n 该变量与脚本被激活时所带的参数相对应。n是正整数,与参数位置相对应($1,$2...)
$? 前一个命令执行后的退出状
$# 提供脚本的参数号
$* 所有这些参数都被双引号引住。若一个脚本接收两个参数,$*等于$1$2
$0 正在被执行命令的名字。对于shell脚本而言,这是被激活命令的路径
$@ 所有这些参数都分别被双引号引住。若一个脚本接收到两个参数,$@等价于$1$2
$$ 当前shell的进程号。对于shell脚本,这是其正在执行时的进程ID
$! 前一个后台命令的进程号
2.2 普通变量
- 赋值:var=value
获取字符串的长度。len=${#var}
- 数值运算:let
let命令后面的变量不用带$,如:
nu=10;
let nu+=10; #nu=20
但这个命令不能进行浮点数的运算。 - 浮点数运算:bc
echo "4 * 0.6" | bc
bc是一个强大的计算器,还可以进项如下操作:
设定小数精度,scale=2,eg:echo "scale=2;3 / 8" | bc
#.37 这是bc的特性,小于0的数,是不显示小数点前的0的。
进制转换。用ibase设定输入数字的进制,obase设定输出数字的进制。
no=10
echo "obase=2;ibase=10;$no" | bc #1010
计算平方以及平方根。
echo "10^4" | bc #1000 平方
echo "sqrt(100)" | bc #10 平方根
2.3 IFS
全称是Internal Field Separtor,内部分隔符。
Shell 的环境变量分为 set, env 两种,其中 set 变量可以通过 export 工具导入到 env 变量中。其中,set 是显示设置shell变量,仅在本 shell 中有效;env 是显示设置用户环境变量 ,仅在当前会话中有效。换句话说,set 变量里包含了 env 变量,但 set 变量不一定都是 env 变量。这两种变量不同之处在于变量的作用域不同。显然,env 变量的作用域要大些,它可以在 subshell 中使用。
而 IFS 是一种 set 变量,当 shell 处理"命令替换"和"参数替换"时,shell 根据 IFS 的值,默认是 space, tab, newline 来拆解读入的变量,然后对特殊字符进行处理,最后重新组合赋值给该变量。
eg:
$ cat test.txt
1
2
3
$ out=$(cat test.txt)
$ echo $out
1 2 3 #shell将(cat test.txt)的结果拆解,并用默认的分隔符(空格)重新组合,赋值给out,因此echo $out的结果不包含换行。
如果要保留cat test.txt中的换行符,一般情况下要做两步:
1是,设定IFS为换行:IFS='\n'
2是,将$(cat test.txt)用双引号引起来,表示不用
若指定IFS为换行符。
2.4 UID
特殊的环境变量,如果UID=0,表示当前以root用户运行脚本。否则不是root
3.自增
Linux Shell中写循环时,常常要用到变量的自增,现在总结一下整型变量自增的方法。
1) i=`expr $i + 1`;
2) let i+=1;
3) ((i++)); #双括号结构
http://www.cnblogs.com/chengmo/archive/2010/10/19/1855577.html
4) i=$[$i+1];
5) i=$(( $i + 1 ))
4.双括号结构(())
双括号结构是对shell中算数及赋值运算的扩展。
语法:
((表达式1,表达式2…))
特点:
- 在双括号结构中,所有表达式可以像c语言一样,如:a++,b--等。
- 在双括号结构中,所有变量可以不加入:“$”符号前缀。
- 双括号可以进行逻辑运算,四则运算.eg. echo $((a>1?2:3));注意四则运算中仍然不支持浮点数运算
- 支持多个表达式运算,各个表达式之间用“,”分开. eg:((a+1,b++,c++))
- 双括号结构 扩展了for,while,if条件测试运算
5.数组
- 取数组长度 – '#'
arr=(1 2 3 4 5)
len=${#arr[@]}
- 打印特定索引的数组元素
echo ${arr[2]} #2
- 打印出数组中的所有值-'*'、'@'
echo ${arr[*]}
echo ${arr[@]}
6.关联数组
在关联数组中,可以用任意的文本作为数组索引。先声明才能使用
- 声明一个关联数组。
declare –A ass_array
- 赋值:
a) ass_array=([index1]=val1 [index2]=val2)
b) ass_array[index1]=val1
ass_array[index2]=val2
echo ${ass_array[index1]}
- 列出数组索引:
echo ${!ass_array[@]}
7.临时文件或目录
在shell脚本中经常要保存临时的数据,如果使用认为创建临时文件用户保存临时数据,则有可能出现重名的情况,导致覆盖原来的数据。
mktemp prefile.xxx 创建以prefile开头的随机文件文件,并返回文件名,指定前缀时必须包含至少3个xxx。
主要参数:
-d : 创建一个目录,dirname=mktemp -d
-u : 仅生成随机文件名,但不创建实际的文件或目录,tmpfile=mktemp -u
原文链接:http://blog.keepmovingxin.com/2016/05/02/Learn-shell/