Syntax error: Bad for loop variable引出的bash和dash的区别
1 问题描述
能在一个linux服务器上运行的sh脚本,拷贝到个人pc ubuntu20.04 LTS上却不能正常运行,sh脚本为:
#!/bin/sh
a="area"
b=".txt"
for ((i=1; i<2; ++i));do
filename="$a$i$b"
echo $filename
done
提示的错误为:
$ ./test.sh
./test.sh: 4: Syntax error: Bad for loop variable
提示是循环变量出了问题。
2 原因分析
从Ubuntu 6.10开始,将先前默认使用bash(the GNUBourne-Again Shell更换到了dash(theDebian Almquist Shell)。
其表现为 /bin/sh 链接到了/bin/dash而不是传统的/bin/bash,可以通过以下命令查看
$ ls -l /bin/sh
lrwxrwxrwx 1 root root 4 Apr 23 10:05 /bin/sh -> dash
但Login Shell还是bash。 原因是dash更快、更高效,而且它符合POSIX规范。Ubuntu在启动的时候会运行很多shell脚本,使用dash可以加快启动速度。
使用的sh脚本声明使用的/bin/sh,所以在使用sh执行检测的时候实际使用的是dash,而dash不支持这种C语言格式的for循环写法。
3 解决方案
3.1 将默认shell更改为bash(bash支持C语言格式的for循环)
执行
$ sudo dpkg-reconfigure dash
出现
选择No,按Enter,会提示:
$ sudo dpkg-reconfigure dash
Removing 'diversion of /bin/sh to /bin/sh.distrib by dash'
Adding 'diversion of /bin/sh to /bin/sh.distrib by bash'
Removing 'diversion of /usr/share/man/man1/sh.1.gz to /usr/share/man/man1/sh.distrib.1.gz by dash'
Adding 'diversion of /usr/share/man/man1/sh.1.gz to /usr/share/man/man1/sh.distrib.1.gz by bash'
再次执行sh脚本,即可运行。
若想切换为默认dash,则选择Yes,按Enter,会提示:
Removing 'diversion of /bin/sh to /bin/sh.distrib by bash'
Adding 'diversion of /bin/sh to /bin/sh.distrib by dash'
Removing 'diversion of /usr/share/man/man1/sh.1.gz to /usr/share/man/man1/sh.distrib.1.gz by bash'
Adding 'diversion of /usr/share/man/man1/sh.1.gz to /usr/share/man/man1/sh.distrib.1.gz by dash'
3.2 直接使用bash检测
在执行sh脚本的时候,指定使用bash来执行,可以在提交时执行:
$ /bin/bash test.sh
即可顺利执行,
3.3 修改循环变量的写法
为了确保shell脚本的可移植性,可以直接更改shell脚本,使用shell支持的for循环格式:
for a in `seq $num`
类似的写法。
3.4 修改sh脚本声明
将sh脚本的
#!/bin/sh
修改为
#!/bin/bash
再次执行
$ ./test.sh
即可顺利运行。
4 常见shell类型及区别
转载自:http://www.happycxz.com/m/?p=137
4.1 Bourne shell (sh)
UNIX 最初使用,且在每种 UNIX 上都可以使用。
在 shell 编程方面相当优秀,但在处理与用户的交互方面做得不如其他几种shell。
4.2 C shell (csh)
csh, the C shell, is a command interpreter with a syntax similar to the C programming language.
一个语法上接近于C语言的shell。
4.2 Korn shell (ksh)
完全向上兼容 Bourne shell 并包含了 C shell 的很多特性。
4.3 Bourne Again shell (bash)
Linux 操作系统缺省的 shell。
是 Bourne shell 的扩展,与 Bourne shell 完全向后兼容。
在Bourne shell 的基础上增加、增强了很多特性。
可以提供如命令补全、命令编辑和命令历史表等功能。
包含了很多 C shell 和 Korn shell 中的优点,有灵活和强大的编程接口,同时又有很友好的用户界面。
4.4 Debian Almquist Shell(dash)
原来bash是GNU/Linux 操作系统中的 /bin/sh 的符号连接,但由于bash过于复杂,有人把 bash 从 NetBSD 移植到 Linux 并更名为 dash,且/bin/sh符号连接到dash。
Dash Shell 比 Bash Shell 小的多(ubuntu16.04上,bash大概1M,dash只有150K),符合POSIX标准。
Ubuntu 6.10开始默认是Dash。
5 规范和建议
标记为# !/bin/sh
的脚本不应使用任何 POSIX 没有规定的特性 (如 let 等命令, 但# !/bin/bash
可以)。
bash支持的写法比dash(ubuntu中的sh)多很多。
想要支持sh xx.sh
运行的,必须遵照 POSIX 规范去写。
想要脚本写法多样化,不需要考虑效率的,可以将文件头定义为# !/bin/bash
, 而且不要使用 sh xx.sh
这种运行方式。