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

bash, sh, dash 傻傻分不清楚

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这种运行方式。

6 Dash与Bash的语法区别

参考:Dash与Bash的语法区别

bash, sh, dash 傻傻分不清楚

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值