目录
2.1 什么是变量
2.2变量的命名
2.3 变量的类型
2.3.1 根据数据类型分类
2.3.2 根据作用域分类
2.4 变量的定义
2.5 shell中的引用
在程序设计语言中,变量是一个非常重要的概念。也是初学者在进行Shell程序设计之前必须掌握的一个非常基础的概念。只有理解变量的使用方法,才能设计出良好的程序。本节将介绍Shell中变量的相关知识
2.1 什么是变量
顾名思义,变量就是程序设计语言中的一个可以变化的量,当然,可以变化的是变量的值。几乎所有的程序设计语言中都有定义变量,并且其涵义也大同小异。从本质上讲,变量就是在程序中保存用户数据的一块内存空间,而变量名就是这块内存空间的地址。
在程序的执行过程中,保存数据的内存空间的内容可能会不断地发生变化,但是,代表内存地址的变量名却保持不变。
2.2变量的命名
在Shell中,变量名可以由字母、数字或者下划线组成,并且只能以字母或者下划线开头。对于变量名的长度,Shell并没有做出明确的规定。因此,用户可以使用任意长度的字符串来作为变量名。但是,为了提高程序的可读性,建议用户使用相对较短的字符串作为变量名。
在一个设计良好的程序中,变量的命名有着非常大的学问。通常情况下,用户应该尽可能选择有明确意义的英文单词作为变量名,尽量避免使用拼音或者毫无意义的字符串作为变量名。这样的话,用户通过变量名就可以了解该变量的作用。
2.3 变量的类型
2.3.1 根据数据类型分类
Shell是一种动态类型语言和弱类型语言,即在Shell中,变量的数据类型毋需显示地声明,变量的数据类型会根据不同的操作有所变化。准确地讲,Shell中的变量是不分数据类型的,统一地按照字符串存储。但是根据变量的上下文环境,允许程序执行一些不同的操作,例如字符串的比较和整数的加减等等。
什么是弱类型语言、强类型语言?
强类型和弱类型主要是站在变量类型处理的角度进行分类的。强类型是指不允许隐式变量类型转换,弱类型则允许隐式类型转换。
(1)强类型语言,当你定义一个变量是某个类型,如果不经过代码显式转换(强制转化)
过,它就永远都是这个类型,如果把它当做其他类型来用,就会报错
(2)弱类型语言,你想把这个变量当做什么类型来用,就当做什么类型来用,语言的解析器
会自动(隐式)转换。
比如:
C语言定义变量,int+变量名,实则前面的int就是给变量内存划分了等级,int定义整型
所以空间里只能存放整型,这就是强类型。php、C#和Python等都是强类型语言。
可以使用declare定义变量的类型:declare attribute variable
注:declare命令还可输出所有的变量、函数、整数和已经导出的变量
+/- "-"可用来指定变量的属性,"+"则是取消变量所设的属性
-p:显示所有变量的值
-i:将变量定义为整数,在之后就可以直接对表达式求值,结果只能是整数。如果求值失败或者不是整数,就设置为0。
-r:将变量声明为只读变量。只读变量不允许修改,也不允许删除。(也可使用readonly定义只读变量)
-a:变量声明为数组变量。但这没有必要,所有变量都不必显示定义就可以用作数组。事实上,在某种意义上,似乎所有变量都是数组,而且赋值给没有下标的变量与赋值给下标为0的数组元素相同。
-f:显示所有自定义函数,包括名称和函数体。
-x:将变量设置成环境变量。可使用+x将变量变成非环境变量
#声明整数型变量
[root@kittod ~]# declare -i ab
[root@kittod ~]# ab=33
[root@kittod ~]# echo $ab
33
#改变变量属性
[root@kittod ~]# declare -i ef
[root@kittod ~]# ef=1
[root@kittod ~]# echo $ef
1
[root@kittod ~]# ef="wer"
[root@kittod ~]# echo $ef
0
[root@kittod ~]# declare +i ef
[root@kittod ~]# ef="wer"
[root@kittod ~]# echo $ef
wer
#设置变量只读
[root@kittod ~]# declare -r ab
[root@kittod ~]# ab=22
-bash: ab: readonly variable
[root@kittod ~]# echo $ab
33
#声明数组变量
[root@kittod ~]# declare -a cd='([0]="a" [1]="b" [2]="c")'
[root@kittod ~]# echo ${cd[1]}
b
[root@kittod ~]# echo ${cd[@]}
a b c
2.3.2 根据作用域分类
根据作用域可将变量分为环境变量(全局变量)和普通变量(局部变量)
1、环境变量
环境变量也可称为全局变量,可以在创建它们的shell及其派生出来的任意子进程shell中使用(su - 切换用户会读取新的环境变量),环境变量又可分为自定义环境变量和bash内置的环境变量。
(1)自定义环境变量
一般是指用export内置命令导出的变量,用于定义shell的运行环境,保证shell命令的正确执行。环境变量可以在命令行中设置和创建,但用户退出命令行时这些变量值就会丢失,即该环境变量只在当前shell和子shell中有效。如果希望永久保存环境变量,可以在配置文件中设置。
①用户的环境变量配置(non-login shell)
~/.bash_profile或~/.bashrc
②全局环境变量的配置(login shell)
/etc/bashrc、/etc/profile文件或者/etc/profile.d目录中定义。
注意:按照系统规范,所有环境变量的名字均采用大写形式。在将环境变量应用于用户进程程序之前,都应该用命令export导出。
有一些环境变量,比如HOME,PATH,SHELL,UID,USER等,在用户登录前就已经被/bin/login程序设置好了,通常环境变量被定义并保存在用户家目录下的.bash_profile文件或全局的配置文件/etc/profile中。
(2)bash内置的环境变量
shell内置的环境变量是所有的shell程序都可以使用的变量。shell程序在运行时,都会接收一组变量来确定登录用户名、命令路径、终端类型、登录目录等,这组变量就是环境变量。环境变量会影响到所有的脚本的执行结果。
注:可使用env查看环境变量
(3)普通变量也可称为局部变量,与全局变量相比,局部变量的使用范围较小,通常仅限于某个程序段访问,例如函数内部。在Shell语言中,可以在函数内部通过local关键字定义局部变量,另外,函数的参数也是局部变量。
示例:全局变量和局部变量的区别
#先自定义一下vim编辑器
[root@kittod ~]# cat .vimrc
#按下 Tab 键时,Vim 显示的空格数。
set tabstop=4
#由于 Tab 键在不同的编辑器缩进不一致,该设置自动将 Tab 转为空格。
set expandtab
#Tab 转为多少个空格。
set softtabstop=4
[root@kittod ~]# vim vartest.sh
#!/bin/bash
#定义函数
func()
{
#输出全局变量v1的值
echo "global variable v1 is $v1"
#定义局部变量v1
local v1=2
#输出局部变量v1的值
echo "local variable v1 is $v1"
}
#定义全局变量v1
v1=1
#调用函数
func
#输出全局变量v1的值
echo "global variable v1 is $v1"
[root@kittod ~]# chmod a+x vartest.sh
[root@kittod ~]# ./vartest.sh
global var v1 is 1
local var v1 is 2
global var v1 is 1
2.4 变量的定义
在Shell中,通常情况下用户可以直接使用变量,而毋需先进行定义,当用户第一次使用某个变量名时,实际上就同时定义了这个变量,在变量的作用域内,用户都可以使用该变量。
1、变量定义示例:变量名=变量值
#定义变量a
a=1
#定义变量b
b="hello"
#定义变量c
c="hello world"
#定义备份路径
bak_dir=/data/backup
#把一个命令的结果赋值变量
变量名=`ls`
变量名=$(ls)
注意:
(1)“=”前后不能有空格
[root@localhost ~]# a= 3
-bash: 3: 未找到命令
[root@localhost ~]# b =4
-bash: b: 未找到命令
(2)字符串类型建议用引号括起来,尤其是特殊字符或有空格
stu_name="zhang san"
引用变量:$变量名 或者${变量名}
查看变量:echo $变量名,set(可查看所有变量:包括自定义变量和环境变量),env显示全局变量,declare输出所有的变量、函数、整数和已经导出的变量。
取消变量:unset 变量名
作用范围:仅在当前shell中有效
注:可使用export指令将变量转换成环境变量
2、位置参数和预定义变量
许多情况下,Shell脚本都需要接收用户的输入,根据用户输入的参数来执行不同的操作。
从命令行传递给Shell脚本的参数又称为位置参数,Shell脚本会根据参数的位置使用不同的位置参数变量读取它们的值。
$# 命令行参数的个数
$n 表示传递给脚本的第n个参数,例如$1表示第一个参数,$2表示第二个参数,
$3 表示第三个参数
$0 当前脚本的名称
$* 以"参数1 参数2 参数3…"的形式返回所有参数的值
$@ 以“参数1”“参数2”“参数3”…的形式返回所有的值
$? 前一个命令或者函数的返回状态码。$?的返回值用法:(1)判断命令、脚本或函数等程序是否执行成功(2)若在脚本中调用执行“exit 数字”,则会返回这个数字给 “$?”变量(3)如果是在函数里,则通过“return 数字”把这个数字以函数值的形式传给“$?”
$$ 返回本程序的进程ID(PID),不常用
$! 获取上一个在后台工作的进程的进程号,不常用
$_ 保存之前执行的命令的最后一个参数,不常用
示例:
[root@kittod ~]# cat param.sh
echo "第1个位置参数是$1"
echo "第2个位置参数是$2"
echo "所有参数是: $*"
echo "所有参数是: $@"
echo "参数的个数是: $#"
echo "当前进程的PID是: $$"
[root@localhost test]# bash param.sh shuju1 shuju2
第1个位置参数是shuju1
第2个位置参数是shuju2
所有参数是: shuju1 shuju2
所有参数是: shuju1 shuju2
参数的个数是: 2
当前进程的PID是: 23278
$?的示例:
[root@kittod ~]# cat ping.sh
#!/bin/bash
ping -c2 $1 &> /dev/null
if [ $? = 0 ];then
echo "host $1 is ok"
else
echo "host $1 is fail"
fi
[root@kittod ~]# chmod a+x ping.sh
[root@kittod ~]# ./ping.sh 192.168.217.128
host 192.168.217.128 is ok
[root@kittod ~]# ./ping.sh 192.168.217.129
host 192.168.217.129 is fail
$*和$@的区别示例:
[root@kittod ~]# set -- "i have" a cat
[root@kittod ~]# echo $#
3
[root@kittod ~]# echo $*
i have a cat
[root@kittod ~]# echo $@
i have a cat
[root@kittod ~]# for i in $*;do echo $i;done
i
have
a
cat
[root@kittod ~]# for i in $@;do echo $i;done
i
have
a
cat
[root@kittod ~]# for i in "$*";do echo $i;done
i have a cat
[root@kittod ~]# for i in "$@";do echo $i;done
i have
a
cat
[root@kittod ~]# shift
[root@kittod ~]# echo $1
a
[root@kittod ~]# shift
[root@kittod ~]# echo $1
cat
2.5 shell中的引用
在bash中有很多特殊字符,这些特殊字符就具有特殊含义。引用就是通知shell将这些特殊字符当作普通字符来处理。
说明:反引号中的字符串将被解释为shell命令
[root@kittod ~]# echo "current user is: $USER"
current user is: root
[root@kittod ~]# echo 'current user is: $USER'
current user is: $USER
[root@kittod ~]# echo "current user is: `whoami`"
current user is: root
[root@kittod ~]# echo 'current user is: `whoami`'
current user is: `whoami`