#!/bin/bash和#!/bin/sh
基本介绍
/bin/sh 是dash
/bin/bash 是bash
Dash 由 NetBSD版本的Almquist shell (ash)发展而来,于1997年,由赫伯特·许(Herbert Xu)移植到Linux上,于2002年改名为 dash (Debian Almquist Shell),并建议将 /bin/sh 指向它,以获得更快的脚本执行速度, 是一种 Unix shell。Dash Shell 比 Bash Shell 小的多,符合POSIX标准。只需要较少的磁盘空间,但是它的对话性功能也较少。
Linux 中的 shell 有很多类型,其中最常用的几种是: Bourne shell (sh)、C shell (csh) 和 Korn shell (ksh), 各有优缺点。Bourne shell 是 UNIX 最初使用的 shell,并且在每种 UNIX 上都可以使用, 在 shell 编程方面相当优秀,但在处理与用户的交互方面做得不如其他几种shell。Linux 操作系统缺省的 shell 是Bourne Again shell,它是 Bourne shell 的扩展,简称 Bash,与 Bourne shell 完全向后兼容,并且在Bourne shell 的基础上增加、增强了很多特性。Bash放在/bin/bash中,它有许多特色,可以提供如命令补全、命令编辑和命令历史表等功能,它还包含了很多 C shell 和 Korn shell 中的优点,有灵活和强大的编程接口,同时又有很友好的用户界面。
root@ubuntu:/bin#ls -l sh bash
-rwxr-xr-x 1 root root 1183448 Jun 18 2020 bash
lrwxrwxrwx 1 root root 4 Jan 11 2021 sh -> dash
root@ubuntu:/bin# ls -l bash dash
-rwxr-xr-x 1 root root 1183448 Jun 18 2020 bash
-rwxr-xr-x 1 root root 129816 Jul 18 2019 dash
dash 比 bash 小一个数量级
应该说, /bin/sh 与 /bin/bash 虽然大体上没什么区别, 但仍存在不同的标准. 标记为 “#!/bin/sh” 的脚本不应使用任何 POSIX 没有规定的特性 (如 let 等命令, 但 “#!/bin/bash” 可以). Debian 曾经采用 /bin/bash 更改 /bin/dash,目的使用更少的磁盘空间、提供较少的功能、获取更快的速度。但是后来经过 shell 脚本测试存在运行问题。因为原先在 bash shell 下可以运行的 shell script (shell 脚本),在 /bin/sh 下还是会出现一些意想不到的问题,不是100%的兼用。
如:
1.sh
a=12345
let "a += 1"
echo "a = $a"
b=${a/23/BB}
echo "b = $b"
root@ubuntu:~/test_shell# sh 1.sh
1.sh: 3: let: not found
a = 123456
1.sh: 6: Bad substitution
root@ubuntu:~/test_shell# bash 1.sh
a = 123457
b = 1BB457
注: b=${a/23/BB} 把变量a中的23(仅限第一次出现)替换成BB, 并赋值给 b.
shell脚本中的#!/bin/bash和#!/bin/sh
#!/bin/sh是指此脚本使用/bin/dash来解释执行。
#!/bin/bash是指此脚本使用/bin/bash来解释执行。
其中,#!是一个特殊的表示符,其后,跟着解释此脚本的shell路径。
bash只是shell的一种,还有很多其它shell,如:sh,csh,ksh,tcsh,...
我们可以通过以下一个示例来进行实验,了解#!/bin/bash的使用。
除第一行外,脚本中所有以“#”开头的行都是注释。
1)#!/bin/bash只能放在第一行,如果后面还有#!,那么只能看成是注释。
这里有三个脚本(脚本都要使用”chmod +x scriptname“命令来获得可执行权限):
1.sh:
#!/bin/sh
source abc
echo "hello abc"
2.sh:
#!/bin/bash
source abc
echo "hello abc"
3.sh:
source abc
echo "hello abc"
三个脚本执行的结果:
root@ubuntu:~/test_shell# cat 1.sh
#!/bin/sh
source abc
echo "hello abc"
root@ubuntu:~/test_shell# cat 2.sh
#!/bin/bash
source abc
echo "hello abc"
root@ubuntu:~/test_shell# cat 3.sh
source abc
echo "hello abc"
root@ubuntu:~/test_shell# ./1.sh
./1.sh: 2: source: not found
hello abc
root@ubuntu:~/test_shell# ./2.sh
./2.sh: line 2: abc: No such file or directory
hello abc
root@ubuntu:~/test_shell# ./3.sh
./3.sh: line 1: abc: No such file or directory
hello abc
root@ubuntu:~/test_shell#
可以发现sh中无法运行source命令
其他效果sh和bash一致
如果将1.sh改成:
echo "1.sh"
#!/bin/sh
source abc
echo "hello abc"
那么,执行结果是:
root@ubuntu:~/test_shell# vim 1.sh
root@ubuntu:~/test_shell# cat 1.sh
echo "1.sh"
#!/bin/sh
source abc
echo "hello abc"
root@ubuntu:~/test_shell# ./1.sh
1.sh
./1.sh: line 3: abc: No such file or directory
hello abc
可以发现source命令可以运行了,使用的是bash解释的。
也就是说,脚本忽略了第二行"#!/bin/sh",直接使用当前所在的shell(也就是bash)来解释脚本。
当把1.sh改成:
#!/bin/sh
#!/bin/bash
source abc
echo "hello abc"
执行结果为:
root@ubuntu:~/test_shell# vim 1.sh
root@ubuntu:~/test_shell# cat 1.sh
#!/bin/sh
#!/bin/bash
source abc
echo "hello abc"
root@ubuntu:~/test_shell# ./1.sh
./1.sh: 3: source: not found
hello abc
root@ubuntu:~/test_shell#
source命令无法使用。
说明,#!/bin/sh这一行起到作用了,但#!/bin/bash并没有起作用。
即,在脚本中,除第一行外,脚本中所有以“#”开头的行都是注释。
2)#!后面的路径一定要正确,不正确会报错。
假如,我们把tbash1.sh中第一行的#!后面加了一个不存在的路径"/home/sh":
#!/home/bash
source abc
echo "hello abc"
执行结果为:
root@ubuntu:~/test_shell# vim 1.sh
root@ubuntu:~/test_shell# cat 1.sh
#!/home/bash
source abc
echo "hello abc"
root@ubuntu:~/test_shell# ./1.sh
bash: ./1.sh: /home/bash: bad interpreter: No such file or directory
root@ubuntu:~/test_shell#
系统会提示/home/bash的路径不存在。
3)如果一个脚本在第一行没有加上#!+shell路径这一行,那么,脚本会默认当前用户登录的shell,为脚本解释器。
因此,大家应该养成脚本首行加上#!+shell路径的习惯。
4)/bin/sh不相当于/bin/bash --posix
在网上搜索到的一些博客有人说上边两个相等,但是我在我的ubuntu20.04上测试的并不相等。
我们将脚本2.sh改为:
#!/bin/bash --posix
source abc
echo "hello abc"
执行结果:
root@ubuntu:~/test_shell# vim 1.sh
root@ubuntu:~/test_shell# vim 2.sh
root@ubuntu:~/test_shell# cat 1.sh
#!/bin/sh
source abc
echo "hello abc"
root@ubuntu:~/test_shell# cat 2.sh
#!/bin/bash --posix
source abc
echo "hello abc"
root@ubuntu:~/test_shell# ./1.sh
./1.sh: 2: source: not found
hello abc
root@ubuntu:~/test_shell# ./2.sh
./2.sh: line 2: source: abc: file not found
root@ubuntu:~/test_shell#
可以发现此时bash的表现和sh不一致
sh是无法使用source命令,但这里可以使用source命令。
这里提示没有abc文件,并且不执行下一句echo命令了。
意思应该是按照posix标准解释脚本。
遇到错误后之和的就不执行接下来的命令了。