shell 的变量作用域可分为三种:
局部变量:用关键字 local 修饰,一般用在自定义函数体中,在函数体中有效。
全局变量:在shell中 不加修饰符定义的变量,都是全局变量,在当前进程/脚本中有效
环境变量:在当前进程及其子进程中有效。当打卡shell会话窗口的时候,其实相当于打开了一个进程,而我们通过shell会话窗口的命令行执行的所有命令,都相当于是启动了一个子进程而已。所以当你在命令行用export设置环境变量的时候,该变量对该窗口下的所有程序有效,而当打开新的会话窗口的时候之前export设置的环境变量则失效了,因为已经是一个新的进程了。为什么/etc/profile里面的export设置的环境变量是永久的,因为从linux系统启动,到任何用户登录,都会提前执行/etc/profile文件,所以最顶层的会话进程已经执行过里面的命令,下面的子进程自然有效。
局部变量:
# 定义函数
function testfunc(){
# 局部变量
local a='local_var'
printf "IN testfunc:a=${a}\n"
}
#声明函数
export -f testfunc ;
#执行语句
testfunc ;
printf "OUT testfunc:a=${a}\n"
/* 结果如下:
IN testfunc:a=local_var
OUT testfunc:a=
方法体外,局部变量a不可见 */
全局变量:
# 全局变量
b='global_var'
# 定义函数
function testfunc(){
# 函数体内也可以定义全局变量
c='global_var'
printf "IN testfunc:b=${b}\n"
# 当局部变量与全局变量同名时,函数体内优先局部变量
local b='local_var'
printf "IN testfunc:b=${b}\n"
}
# 声明函数
export -f testfunc ;
# 执行语句
testfunc ;
printf "OUT testfunc:c=${c}\n"
/* 结果如下:
IN testfunc:b=global_var
IN testfunc:b=local_var
OUT testfunc:c=global_var
不加local修饰符定义的都是全局变量,当局部变量与全局变量同名时,函数体内优先局部变量*/
环境变量:
有两个shell脚本程序,a.sh 和 a_sub.sh,我们分别在两个脚本里面定义全局变量与环境变量,然后用 a.sh 调用并执行 a_sub.sh ,测试两个脚本中的变量能否相互访问。
a_sub.sh 代码如下:
# 环境变量
export sub_envi='sub_envi'
# 全局变量
sub_global='sub_global'
# 调用父进程的变量
printf "I am a_sub.sh: fa_envi=${fa_envi}\n"
printf "I am a_sub.sh: fa_global=${fa_global}\n"
a.sh 代码如下:
# 环境变量
export fa_envi='fa_envi'
# 全局变量
fa_global='fa_global'
# 执行子进程
sh a_sub.sh ;
# 调用子进程的变量
printf "I am father: sub_envi=${sub_envi}\n"
printf "I am father: sub_global=${sub_global}\n"
执行sh a.sh的结果如下:
I am a_sub.sh: fa_envi=fa_envi
I am a_sub.sh: fa_global=
I am father: sub_envi=
I am father: sub_global=
说明:子进程可以调用父进程的环境变量,而父进程不能调用子进程的环境变量。
如果a.sh 想要使用a_sub.sh 里面的变量设置怎么办?
用source 命令可以加载a_sub.sh 程序:source ./a_sub.sh。
source命令的作用,是在当前的shell环境中执行命令,相当于把a_sub.sh的所有代码,放到了a.sh文件中一起执行,此时a.sh 和 a_sub.sh 不在是父子进程调用的关系。