问题复现
执行安装sentry的脚本时报错
[root@localhost ~/sentry]# sh install.sh
./install/_lib.sh:行9: 未预期的符号 `>' 附近有语法错误
说这个路径下的一个脚本有格式错误,我们 vim -b 看一下
我们发现并没有语法错误
排查定位
既然sh执行脚本报错,还有其他可以执行脚本的方式,我们都尝试一下
bash install.sh
./install.sh
/root/sentry/install.sh
我们发现用bash和在当前目录下执行脚本以及使用绝对路径执行脚本没有报错
我们知道,使用绝对路径来执行脚本或者在当前目录下执行脚本都是脚本自己本身作为执行文件来执行,所以我们使用这两个方式来执行脚本时需要给脚本添加执行权限
chmod +x test.sh
而使用sh命令或者bash命令执行脚本则是将.sh文件作为参数传递给sh(bash)命令来执行的
这时不是.sh自己来执行,而是被人家调用执行,所以不要执行权限,也不需要指定bash路径
排查到这一步,应该就是sh和bash之间的区别导致脚本能不能成功运行
我们先来看下sh和bash命令的区别
shell 作为一种脚本语言,需要相应的shell编译器进行编译。
sh是UNIX的标准编译器,它具备极好的编译性能,很适合我们进行shell编程,但是它与用户的交互不太好;
bash是目前Linux采用的一款标准的编译器,向后兼容sh,它自身集合sh、ksh等编译器的优点,不仅很适合进行shell编程,同时又具备良好的用户交互,如命令补全、命令转发等优秀交互性能。
(PS:由于bash本身比较复杂,在一些Linux的发行版本中,如Ubunto将其进行了缩减,改为dash,并让/bin/sh指向它)
你用sh编译指向的是bash,你用bash调用指向的仍然是bash,
但是用sh编译时脚本会走posix模式,而用bash编译则不会
[root@localhost ~/sentry]# ll /bin/sh
lrwxrwxrwx. 1 root root 4 7月 11 13:53 /bin/sh -> bash
解决问题
现在我们知道了:
1、使用sh进行编译时相当于bash编译+走posix标准
2、使用bash编译时则没有走posix标准
解决方法:
1、在脚本里面加参数,使得这个脚本符合posix标准
set +o posix
2、使用除sh命令之外的命令进行脚本编译(不走posix标准)
bash install.sh
./install.sh
/root/sentry/install.sh