变量根据作用范围的大小主要分为局部变量
、全局变量
、环境变量
:
(1)局部变量:在Shell脚本中的函数内显示使用local关键字定义的变量。其作用域局限于函数内,当local变量名与global变量名同名时,使用的是local关键字定义的局部变量。
(2)全局变量:一般在Shell脚本的函数外定义,当然也可以在函数内不使用local关键字申明来定义全局变量。 全局变量仅在当前Shell脚本中有效,其他Shell脚本进程不能访问,其作用域从全局变量定义的位置(如果是在函数中定义的全局变量则是从函数被调用时执行变量定义的地方)开始,到脚本结束或被显示删除的地方为止。
name="tom" ## 函数外定义的全局变量name
weight="50kg" ## 函数外定义的全局变量weight
function test(){
local name="jack" ## 函数内定义的局部变量name
sex="man" ## 函数内定义的全局变量sex
age=25 ## 函数内定义的全局变量age
local age=18 ## 函数内定义的局部变量age
age=20 ## 这里其实是重新给local定义的局部变量age重新赋值
echo "test() name is ${name}" ## 对应6中的结果,函数内部定义了局部变量name,所以获取到的是函数内定义的局部变量name的值
echo "test() weight is ${weight}" ## 对应7中的结果,函数内没有定义局部变量weight,所以使用的是全局变量weight的值
echo "test() age is ${age}" ## 对应8中的结果,获取到的是最新赋值后的局部变量值
}
function test2(){
age=30 ## 函数内定义的全局变量age
echo "test2() get sex is: ${sex}"
}
echo "name is ${name}" ## 对应1中的结果拿取的是函数外定义的全局变量name的值
echo "weight is ${weight}" ## 对应2中的结果拿取的是函数外定义的全局变量weight的值
echo "age is: ${age}" ## 对应3中的结果,由于此时两个函数都未被调用,所以获取不到函数中定义的age变量的值
test2 ## 对应4中的结果,由于全局变量sex是在test函数中定义的,此时test函数未被调用,所以获取不到sex变量的值
echo "age is: ${age}" ## 对应5中的结果,由于在test2函数执行后会创建一个全局变量age,所以此时获取到的变量age的值为30
test
echo "age is: ${age}" ## 对应9中的结果,由于test方法执行后重新创建一个全局变量age并赋值为25,所以此处获取到的是重新创建的全局变量age的值
test2 ## 对应10中的结果,由于test函数执行后会创建一个全局变量sex,所以此时再次执行后就可以获取到test函数中创建的全局变量sex的值
echo "age is: ${age}" ## 对应11中的结果,test2函数执行后会重新创建一个值为30的全局变量age,此时再次获取到的是新创建的age的值
(3)环境变量:可以使用export命令行声明,这种属于临时环境变量,只在当前Shell会话中生效,关闭当前Shell会话后失效。永久环境变量需要修改/etc/profile
配置文件,在该文件中定义的变量永久生效。
## 定义一个临时环境变量
export name=tom
## 定义一个永久环境变量
vim /etc/profile
## 进入编辑模式后添加一个想要设置的环境变量后保存退出
world='hello'
## 使用下面命令刷新/etc/profile文件,否则不会立即生效
source /etc/profile
$1,$2,$3 … ${10}, ${11} …
表示,10以后的数字全部用大括号{}括起来,$
数字表示脚本文件后跟的第几个参数。$0
:代表文件本身。$?
:代表上一条命令的结果返回的值,命令执行成功返回0值;失败返回非0值 , return 退出函数时获取的返回值(0-255,超过部分%256取余)。$#
:代表传递到脚本的参数个数。$@
:与$*
相同,但是使用时加引号,并在引号中返回每个参数(结果类似:‘1’ ‘2’ ‘3’)。$*
:代表以一个单字符串显示所有向脚本传递的参数(结果类似:‘1 2 3’)。$$
:代表脚本运行的当前进程ID号。$!
:代表后台运行的最后一个进程的ID号。使用 unset 命令可以删除变量,删除变量后该变量将无法再被使用,unset 命令不能删除只读变量。
使用一个定义过的变量,只要在变量名前面加美元符号即可,变量名外面的花括号是可选的,加不加都行,加花括号是为了帮助解释器识别变量的边界。
name="Tom"
echo $name
echo ${name}
字符串是shell编程中最常用最有用的数据类型,字符串可以用单引号,也可以用双引号,也可以不用引号。
(1) 单引号:
str='hello world'
(2)双引号
name='Tom'
str="Hello, I am \"$name\"! \n"
(3)字符串拼接:下面三种方式输出结果是一样的。
name='Tom'
# 使用双引号拼接
str="hello, "$name" !"
str1="hello, ${name} !"
# 使用单引号拼接
str3='hello, '$name' !'
(4)获取字符串长度
str="abcde"
echo ${#str} # 输出5
(5)提取子字符串
str="hello world"
echo ${str:1:4} # 输出 ello
(6)查找子字符串
str="hello world"
echo `expr index "$str" d` # 输出 11
(1)数组的定义:用括号来表示数组,数组元素用"空格"符号分割开,数组元素的下标由 0 开始编号。下标可以是整数或算术表达式,其值应大于或等于 0,可以不使用连续的下标,而且下标的范围没有限制。
array_name=(value0 value1 value2 value3)
或者
array_name=(
value0
value1
value2
value3
)
(2)读取数组:获取数组中的元素需要利用下标。
array=(1 2 3 4)
# 获取数组中第二个元素
num=${array[1]}
# 获取数组中所有的元素
num=${array[@]} 或者 num=${array[*]}
(3)获取数组的长度
# 取得数组元素的个数
length=${#array[@]} 或者 length=${#array[*]}
# 如果数组下标对应的是一个字符串的话,取得数组中该单个字符串的长度
lengthn=${#array[n]}
Shell支持多种运算符,主要包括:算数运算符、关系运算符、布尔运算符、字符串运算符、文件测试运算符。
+(加)
、-(减)
、*(乘)
、/(除)
、%(取余)
、=(赋值)
、==(等于)
、!=(不等于)
num=`expr 2 + 2`
num=`expr 2 \* 2`
注意:
1.表达式和运算符之间要有空格,例如 2+2 是不对的,必须写成 2 + 2
2.完整的表达式要被 ` ` 包含,注意这个字符不是常用的单引号
3.乘号(*)前边必须加反斜杠(\)才能实现乘法运算,如果是在 MAC 中 shell 的 expr 语法是:$((表达式)),
此处表达式中的 "*" 不需要转义符号 "\"
-eq(等于)
、-ne(不等于)
、-gt(大于)
、-lt(小于)
、-ge(大于等于)
、-le(小于等于)
、==(等于)
、!=(不等于)
、>(大于)
、<(小于)
、>=(大于等于)
、<=(小于等于)
a=2
b=5
注意:
1.使用[]或者[[]]时,变量与[]或者[[]]之间必须要有空格隔开。
if [ $a -lt $b ]
if [[ $a -lt $b ]]
2.使用(())时,变量与(())之间不需要空格隔开,且不能使用英文表示的运算符
if (($a<$b))
if (($a>=$b))
3.使用[]且不使用英文代表的运算符时,>和<前面需要使用转义符\,防止当作重定向符使用,使用[[]]和(())不需要转义
if [ $a \< $b ]
if [[ $a < $b ]]
if (($a<$b))
4.使用英文表示的关系运算符时,运算符与变量之间需要空格隔开,使用符号表示的关系运算符时,
运算符与变量之间最好不要使用空格隔开
if [[ $a <= $b ]] ## 会报错
if [[ $a<=$b ]]
if (($a<=$b))
-a(与)
、-o(或)
、!(非)
a=2
b=2
if ! [ $a -gt $b ]
if ! [ $a \> $b ]
if ! [[ $a -gt $b ]]
if [ $a -lt 20 -a $b -gt 10 ]
if [ $a -lt 30 -o $b -gt 10 ]
注意:
1.使用!运算符时需要与[]之间保持一个空格隔开。
2.!可以在[ ]、[[ ]]中使用,但不可以在(( ))中使用。
3.使用-a或者-o时,必须使用[]。
&&(与)
、||(或)
a=5
b=15
if [[ $a -lt 10 && $b -gt 10 ]] ## 逻辑与,如果左边为真,则继续判断右边表达式的真假,如果左边为假,则整个表达式最终结果为假,就没必要再去判断右边表达式的真假
if [[ $a -lt 10 || $b -gt 10 ]] ## 逻辑或,如果左边为真,则整个表达式最终结果即为真,也就没必要再去判断右边表达式的真假,如果左边为假,还需要继续判断右边的表达式的真假
注意:
1.逻辑运算符必须在[[ ]]或(( ))中才有效,否则报错;
=(相等)
、!=(不相等)
、-z(检测字符串长度是否为0)
、-n(检测字符串长度是否不为 0)
、$(检测字符串是否不为空)
a="abc"
b="efg"
if [ $a = $b ]
if [ $a != $b ]
if [ -z $a ]
if [ -n "$a" ]
if [ $a ]
file="/root/test.sh"
1.-b检测文件是否是块设备文件,如果是,则返回 true。
if [ -b $file ]
2.-c检测文件是否是字符设备文件,如果是,则返回 true。
if [ -c $file ]
3.-d检测文件是否是目录,如果是,则返回 true。
if [ -d $file ]
4.-f检测文件是否是普通文件(既不是目录,也不是设备文件),如果是,则返回 true。
if [ -f $file ]
5.-r检测文件是否可读,如果是,则返回 true。
if [ -r $file ]
1.-w检测文件是否可写,如果是,则返回 true。
if [ -w $file ]
1.-x检测文件是否可执行,如果是,则返回 true。
if [ -x $file ]
1.-s检测文件是否为空(文件大小是否大于0),不为空返回 true。
if [ -s $file ]
1.-e检测文件(包括目录)是否存在,如果是,则返回 true。
if [ -e $file ]
if [ $a == $b ]
then
...
elif [ $a -gt $b ]
then
...
else
...
fi
或者写成一行
if [ "$a" -gt "$b" ]; then ...; fi
如果使用 ((...)) 作为判断语句,大于和小于可以直接使用 > 和 <
if (( $a > $b )); then
...
fi
case in 语句跟 if 语句类似,也是一种条件选择语句,当分支较多,并且判断条件比较简单时,使用 case in 语句就比较方便了
case expression in
pattern1)
...
;;
pattern2)
...
;;
pattern3)
...
;;
……
*)
...
esac
1.expression 既可以是一个变量、一个数字、一个字符串,还可以是一个数学计算表达式,或者是命令的执行结果,
只要能够得到 expression 的值就可以。
2.pattern 可以是一个数字、一个字符串,甚至是一个简单的正则表达式。
3.case 会将 expression 的值与 pattern1、pattern2、pattern3 逐个进行匹配,匹配成功就执行后面对应的所有语句
(该语句可以有一条,也可以有多条),直到遇见双分号 ;; 结束,跳出整个 case 语句。
4. case in 类似Java的 switch ... case 语句,而 ;; 类似其中的break跳出循环的作用,*) 类似其中的 default起到托底作用。
5.最后的 *) 部分可以没有,最后的 *) 部分中的 ;; 可以写也可以不写,但是其他分支的 ;; 必须要写。
1.for 循环中的 exp1(初始化语句)、exp2(判断条件)和 exp3(自增或自减)都是可选项,都可以省略(但分号;必须保留)
for((exp1; exp2; exp3))
do
...
done
例1. 计算从 1 加到 100 的和
for ((i=1; i<=100; i++))
do
((sum += i))
done
2.for in 循环,每次循环都会从list中取出一个值赋给变量 num,然后进入循环体(do 和 done 之间的部分),
执行循环体中的操作。直到取完list中的所有值,循环就结束了。其中list的形式有多种,你可以直接给出具体的一系列值,
也可以给出一个范围,还可以使用命令产生的结果,甚至使用通配符。
for num in list
do
...
done
例2. 计算从 1 加到 100 的和
sum=0
for n in {1..100}
do
((sum+=n))
done
或者写成一行
for n in {1..100}; do ((sum+=n)); done;
while 循环是 Shell 脚本中最简单的一种循环,当条件满足时,while 重复地执行一组语句,当条件不满足时,就退出 while 循环。
while condition
do
command
done
例1.计算从 1 加到 100 的和
i=1
sum=0
while ((i <= 100))
do
((sum += i))
((i++))
done
例2.实现无限循环
while true
do
command
done
或者
while :
do
command
done
ntil 循环与 while 循环在处理方式上刚好相反,循环执行一系列命令直至条件为 true 时停止
until condition
do
command
done
例1.计算从 1 加到 100 的和
i=1
sum=0
until ((i > 100))
do
((sum += i))
((i++))
done
Shell 也是通过 break 和 continue 两个关键字来跳出循环,和Java中的功能是完全一样的,两者的区别跟Java的也是一样的。
break 和 continue 的区别在于 break 是跳出整个循环,而 continue 只是跳出当前循环。
例1. 不断的计算100除以输入的数的结果,直到当输入的值为0时退出循环
num=100
while read n; do
if((n==0)); then
echo "can not input ${n}"
break
else
result=`expr $num / $n`
echo "the result is: ${result}"
fi
done
例2. 不断的计算100除以输入的数的结果,当输入的值为0时,输出一个提示“请输入任意非0的数字”
while read n; do
if((n==0)); then
echo "please input not ${n} number"
continue
else
result=`expr $num / $n`
echo "the result is: ${result}"
fi
done
shell中函数标准的定义格式如下:
function name() {
statements
[return value]
}
也可以不写 function 关键字:
name() {
statements
[return value]
}
如果写了 function 关键字,也可以省略函数名后面的小括号:
function name {
statements
[return value]
}
注意:
1.return value 表示函数的返回值,value 的值是 (0-255之间的任意数,当然也可以不写 return value,
这时会将最后一条命令运行结果,作为返回值。
2.方法的()中没有参数的定义,这个跟Java的方法参数定义不太一样,shell的参数是在调用函数的时候直接写在函数后面的。
## 定义一个函数
function name(){
echo "初始化的name为:TOM !"
echo "接收第一个name参数为 $1 !"
echo "接收第二个name参数为 $2 !"
echo "接收的参数总数有 $# 个!"
echo "作为一个字符串输出所有参数 $* !"
return $#
}
## 不传参调用函数
echo "=============first use with no param=============="
name
## 传参调用函数
echo "=============second use with param================"
name Tom Jack
## 输出函数返回结果
echo "the function result is:$?"