前言:自从大学毕业参加工作以来,接触的开发工作都是在服务端完成,于是接触了比较多的Linux当做开发机使用,或多或少有一些重复性的工作,于是开始琢磨学习一些shell脚本的知识,以便处理这些繁琐的事情。
shell种类有很多bash、sh、csh、ksh等,不同的环境下使用的也不同,不过bash应该是使用最广泛了~也是Linux发行版默认的;
主要就是掌握常见的命令以及语法格式吧~然后就可以上手干活了:
命令如下:
- echo:输出命令到终端
- -e:表示对转义字符进行替换
- -E:禁止转义
- read:从终端中读入命令
- readonly:将命令标记为只读命令
- unset:删除变量,但是不能删除readonly标记的变量
- printf:usage: printf format-string [arguments...] eg:printf "%s %s %s\n" a b c d e f g h i j
转义字符:
- \\:反斜杠
- \a:警报
- \b:退格
- \f:换页
- \n:换行
- \r:回车
- \t:tab键
- \v:垂直制表符
特殊变量:
- $0:当前脚本的文件名
- $n:表示传递给文件或者函数的参数,n是几就表示第几个参数
- $#:表示参数的个数
- $*:表示所有的参数
- $?:上个命令的退出状态或则函数的返回值
- $$:当前shell的进程ID
- $@:也是表示所有参数,和$*的区别就是:"$@"会显示所有的参数,每个一行,"$*"将所有的参数显示在一行
命令替换:`command`
DATE=`date` echo "Date is $DATE"USERS=`who | wc -l` echo "Logged in user are $USERS"UP=`date ; uptime` echo "Uptime is $UP"
变量替换:
- ${var} 变量本来的值
- ${var:-word} 如果变量var为空或则Unset,那么返回word,但是不改变var的值
- ${var:=word}如果变量var为空或则unset,那么返回word,并将var的值设置为word
- ${var:?messgage}如果变量var为空或则unset,那么messgae将送到标准错误输出,脚本停止运行
- ${var:+word}如果变量var被定义,那么返回word,但不改变var的值
运算符:
- 算术运算符(expr):
- + 加法 `expr $a + $b`
- - 减法 `expr $a - $b`
- * 乘法 `expr $a \* $b`
- / 除法 `expr $b / $a`
- % 取余 `expr $b % $a`
- = 赋值 a=$b 把变量b的值赋给a
- == 相等,用于比较两个数字,相同则返回true [$a == $b]
- != 不相等。用于比较两个数字,不相同则返回true [$a != $b]
- eg:
#!/bin/sh a=10 b=20 val=`expr $b % $a` 18.echo "b % a : $val"
- 关系运算符
- -eq 检测两个数是否相等,相等则返回true
- -ne 检测两个数是否不相等,不相等则返回true
- -gt 检测左边的数是否大于右边的数,如果是,则返回true
- -lt 检测左边的数是否小于右边的数,如果是,则返回ture
- -ge 检测左边的数是否大于等于右边的数,如果是,则返回true
- -le 检测右边的数是否小于等于右边的数,如果是,则返回true
- eg:
#!/bin/sh a=10 b=20 if [ $a -eq $b ] then echo "$a -eq $b : a is equal to b" else echo "$a -eq $b: a is not equal to b" fi
- 布尔运算符:
- ! 非运算 [!false] 返回true
- -o 或运算 [$a -lt 20 -o $b -gt 100]
- -a 与运算 [$a -lt 20 -a $b -gt 100]
- 字符串运算符:中括号两边要留个空隙
- = 检测两个字符串是否相等,相等则返回true [ $a = $b ]返回false
- != 检测两个字符串是否相等,不相等返回true [ $a != $b ]返回true
- -z 检测字符串长度是否为0, 为0返回ture [ -z $a ]
- -n 检测字符串长度是否不为0,不为0返回true [ -n $a ]
- str 检测字符串是否为空,不为空返回true [ $a ] 返回true
- 文件测试运算符:
- -b file 检测是否是块设备文件 [-b $file]
- -c file 检测是不是字符设备文件 [-c $file]
- -d file 检测是不是目录 [-d $file]
- -f file 检测是不是普通文件(既不是目录,也不是设备文件) [-f $file]
- -g file 检测是否设置了SGID位
- -k file 检测是否设置了(sticky bit)
- -p file 检测是否具有管道
- -u file 检测是否设置了SUID位
- -r file 检测是否可读
- -w file 检测是否可写
- -x file 检测是否可以执行
- -s file 检测是否为空文件(大小为0)
- -e file 检测文件是否存在
字符串操作:
- ${#string} $string的长度
- ${string:position} 在$string中,从位置$position开始提取子串
- ${string:position:length} 在$string中,从位置$position开始提取长度为$length的子串
- ${string#substring} 从变量$string的开头,删除最短匹配$substring的子串
- ${string##substring} 从变量$string的开头,删除最长匹配$substring的子串
- ${string%substring} 从变量$string的结尾,删除最短匹配$substring的子串
- ${string%%substring} 从变量$string的结尾,删除最长匹配$substring的子串
- ${string/substring/replacement} 在$string中用$replacement取代第一个匹配的$substring
- ${string//substring/replacement} 在$string中用$replacement取代全部匹配的$substrng
- ${string/#substring/replacement} 如果$string的前缀匹配$substring,那么用$replacement取代$substring
- ${string/%substring/replacement} 如果$string的后缀匹配$substring,那么用$replacement取代$substring
数组的操作:
定义数组:array_name=(value0 value1 value2 value3)数组的操作:
读取数组:${array_name[2]}
读取数组的元素的个数:${#array_name[@]}/${#array_name[*]}
返回数组的所有的下标:${!array_name[@]}/${!array_name[*]}
返回数组的所有的元素:${array_name[@]}/${!array_name[*]}
注意的事项:
- 变量的初始化中间不能有空格,eg: name="Spider"
- 变量的使用时候,最好以${name}这种形式,确认变量的边界;eg: echo 'I am ${name}-spiders'
- 注释只能用#号
- if..else使用:
- if then... fi 语句;
- if then... else ... fi 语句;
- if then... elif then... else ... fi 语句。
- case..switch
-
echo 'Input a number between 1 to 4' echo 'Your number is:\c' read aNum case $aNum in 1) echo 'You select 1' ;; 2) echo 'You select 2' ;; 3) echo 'You select 3' ;; 4) echo 'You select 4' ;; *) echo 'You do not select a number between 1 to 4' ;; esac
-
- for循环:
-
for 变量 in 列表 do command1 command2 ... commandN done
-
- while循环:
-
while command do Statement(s) to be executed if command is true done
-
- until循环:
-
until command do Statement(s) to be executed until command is false done
-
- break and continue ,和其它语言不同的是,shell中的可以后加数字n,表示第几层循环
- 函数:
- 注意,$10 不能获取第十个参数,获取第十个参数需要${10}。当n>=10时,需要使用${n}来获取参数
function_name () { list of commands [ return value ] }
command > file 输出重定向到file
- command < file 输入重定向到file
- command >> file 输出以追加的方式重定向到file
- n > file 将文件描述符重定向到file
- n >> file 将文件描述符以追加的方式重定向到file
- n >& m 将输出文件m和n合并
- n <& m 将输入文件m和n合并
- << tag 将开始标记tag和结束标记tag之间的内容作为输入
- 注意,$10 不能获取第十个参数,获取第十个参数需要${10}。当n>=10时,需要使用${n}来获取参数
- 重定向,如果希望将 stdout 和 stderr 合并后重定向到 file,可以这样写:
- $command > file 2>&1
- 重定向到/dev/null 文件,屏蔽输出结果
- 调试shell 脚本:在#! /bin/bash -x 就可以看到脚本执行的顺序
- 通配符
shell常见通配符:
字符 含义 实例 * 匹配 0 或多个字符 a*b a与b之间可以有任意长度的任意字符, 也可以一个也没有, 如aabcb, axyzb, a012b, ab。 ? 匹配任意一个字符 a?b a与b之间必须也只能有一个字符, 可以是任意字符, 如aab, abb, acb, a0b。 [list] 匹配 list 中的任意单一字符 a[xyz]b a与b之间必须也只能有一个字符, 但只能是 x 或 y 或 z, 如: axb, ayb, azb。 [!list] 匹配 除list 中的任意单一字符 a[!0-9]b a与b之间必须也只能有一个字符, 但不能是阿拉伯数字, 如axb, aab, a-b。 [c1-c2] 匹配 c1-c2 中的任意单一字符 如:[0-9] [a-z] a[0-9]b 0与9之间必须也只能有一个字符 如a0b, a1b... a9b。 {string1,string2,...} 匹配 sring1 或 string2 (或更多)其一字符串 a{abc,xyz,123}b a与b之间只能是abc或xyz或123这三个字符串之一。
只知道不行动是纸上谈兵,只行动不思考更像是无头的苍蝇,要知行合一。