变量定义
变量类型 | 声明及赋值方法 | 作用域 |
---|---|---|
局部变量 | 变量名=值 | 仅当前声明变量的脚本内 |
全局变量 | export 变量名=值 | 当前终端环境下执行的所有脚本 |
全局变量在修改值的时候也需要通过 export 变量名=值
使用nvm切换node环境时,已声明的变量依然能有效的被访问,无需重新声明
变量的访问
使用 $变量名
来访问变量的值
局部变量案例
局部变量作用域仅在当前sh脚本内,不会跨越到别的sh脚本内
# 编写一个脚本:main.sh
echo "local=$local"
local="tom"
echo "local=$local"
# 终端内执行main.sh脚本
sh main.sh
# 输出结果:
# local=
# local=tom
全局变量案例
全局变量自声明起在当前终端内会一直生效,可以跨越sh脚本,过程中可以随时变更变量值
# main.sh
echo "main.sh global=$global"
# 调用另一个脚本来看看全局变量是不是作用到内部调用的别的脚本内
sh sub.sh
# sub.sh
echo "sub.sh global=$global"
# 终端内先export定义一个全局变量
export global=tom
# 然后我们执行main.sh,看看shell脚本内能否获取到这个变量
sh main.sh
# 输出结果:
# main.sh global=tom
# sub.sh global=tom
在脚本内部也可以随时利用export声明一个全局变量
# main.sh
echo "main.sh global=$global"
export global=tom
echo "main.sh global=$global"
# 执行一个子脚本
sh sub.sh
# 终端内执行main.sh脚本
sh main.sh
# 输出结果:
# main.sh global=
# main.sh global=tom
# sub.sh global=tom
IF判断
基本用法
if [ 表达式 ]
then
echo "逻辑true"
else
echo "逻辑false"
fi
多个if else
if [ 表达式1 ]
then
echo "1"
elif [ 表达式2 ]
then
echo "3"
elif [ 表达式3 ]
then
echo "3"
else
echo "unknow"
fi
字符判断
# 相等判断
if [ $global = "watch" ]
then
echo "true"
else
echo "false"
fi
# 非空值判断
if [ -n "$global" ]
# 空值判断
if [ ! -n "$global" ]
# 不等判断
if [ $global != "watch" ]
# 变量对比
if [ $global = $local ]
# 可靠的值判断
global=tom
if [ $test = $global ]
then
echo "true"
fi
# 这里执行会报错 [: =: unary operator expected
# 这是由于 test 变量根本没定义,所以导致这条语句变为了 if [ = tom ] 语法错误了
# 解决方法一:用可靠的等值判断
if [ "$test"x = "$global"x ]
then
echo "true"
fi
# 这里将被翻译为 if [ x = tomx ] 所以语法没有问题,也不影响值得判断
提示
IF语句后面有时可以看到[],有时会是[[]],区别是[[]]是bash支持的扩展语法,只有在bash中可以使用,[]是标准语法可移植性更强
[[]]扩展支持==、!=、<、>、<=、>=、=~等,而[]只支持=和!=
if语句内
=
两侧要留一个空格
逻辑与和逻辑或
使用 -a
和 -o
在一个if语句内连接
name=tom
age=20
# 逻辑与用 -a 连接左右两边等式
if [ $name = "tom" -a age = 20 ]
# 逻辑或用 -o 连接左右两边等式
if [ $name = "jerry" -o $name = "tom" ]
使用 &&
和 ||
name=tom
age=20
# 逻辑与
if [ $name = "tom" ] && [ age = 20 ]
# 逻辑或
if [ $name = "jerry" ] || [ $name = "tom" ]
数值判断
只能对数值型的变量进行判断,字符串会引发异常
# 大于判断
if [ $global -gt 100 ]
# 大于等于判断
if [ $global -ge 100 ]
# 小于判断
if [ $global -lt 100 ]
# 小于等于判断
if [ $global -le 100 ]
# 不等于判断
if [ $global -ne 100 ]
等待用户输入一个值给变量
read -p "请选择环境:1、预发测试 2、生产:" build_env
确定nvm命令所在路径
if [ -d "${HOME}/.nvm/" ]
then
. ${HOME}/.nvm/nvm.sh
fi
判断yarn是否安装
ss=`yarn -v`
flag=$?
if [ $flag != 0 ]
then
echo "yarn没有安装"
fi
判断node是否为某个版本
for line in `node -v`
do
if [ $line != "v10.15.0" ]
then
echo "node不是10.15.0"
fi
done
判断某个方法返回值
checkNode(){
return 1
}
checkNode
ret=$?
if [ $ret = 1 ]
then
echo "checkNode返回值是1"
fi
sh、bash、source的区别
sh是shell程序
bash是增强版的sh
source是shell程序内的一个命令,简单理解就是在一个.sh文件内可以使用source命令,而直接在终端内可能某些情况下是无法直接执行source的
sh和source在执行shell脚本时的区别
sh 命令会启动一个新的 sh 进程来执行指定的脚本,并将脚本的输出返回给父进程。这意味着,在子进程中定义的变量和函数不会传递到父进程中。
source 命令会在当前 Shell 环境中执行指定的脚本,而不是启动一个新的 Shell 进程。这意味着,在执行 source 命令时,脚本中定义的变量和函数会在当前 Shell 环境中生效,可以被后续的命令和脚本使用
例如:
# task1.sh
export env=dev
env2=prod
sh task2.sh #输出 env=dev和env2=,在windows的gitbash下输出 env= 和 env2=
source task2.sh #输出 env=dev和env2=prod
# task2.sh
echo env=$env
echo env2=$env2
脚本内调用 sh a.sh和sh ./a.sh的区别
sh a.sh
执行时,会优先到系统环境变量$PATH的所有目录下寻找有没有a.sh,有则执行
sh ./a.sh
执行时,会直接在当前第一个shell执行的所在目录下寻找a.sh
假如你的文件目录结构如下
\user\
a.sh
shells\
b.sh
c.sh
a.sh内容为 sh ./shells/b.sh
b.sh内容为:
sh ./c.sh
当你在\user目录下执行sh a.sh
时,会发现报错,找不到c.sh,那是因为sh ./c.sh
寻找文件的根目录是以第一个脚本启动时 a.sh
所在目录为根目录的
如果你的脚本想要的效果是寻找当前本脚本目录下的别的脚本则可以如下解决:
# 获取当前脚本执行的目录
SCRIPT_DIR="$(cd "$(dirname "$0")" && pwd)"
# 通过绝对路径来调用 b.sh
. "$SCRIPT_DIR/c.sh"