我有一个名为DS1. 我想4次复制,所以我也有DS2,DS3,DS4和DS5。
我可以从终端使用以下命令并且它有效:
for i in {2..5}; do cp -r /home/test/DS1 "DS$i"; done
Run Code Online (Sandbox Code Playgroud)
但是,当我将其输入到这样的脚本中时:
#!/bin/sh
for i in {2..5}; do cp -r /home/test/DS1 "DS$i"; done
Run Code Online (Sandbox Code Playgroud)
它创建一个名为DS{2..5}的目录,而不是增量介于 2 和 5 之间的目录。不明白我做错了什么。
{2..5}是大括号扩展。POSIX没有标准化大括号扩展。一些(但不是全部)广泛使用的Bourne 风格的shell 支持它。
您在 Ubuntu 上的终端中与之交互的 shell 是bash,除非您故意使用不同的 shell 。bash支持大括号扩展。但是家当你的脚本是sh,它在Ubuntu是一个符号链接到dash。dash不支持大括号扩展。
所以你可以:
bash脚本(或其他支持大括号扩展的 shell 的脚本,例如zsh或ksh)。dash.如果您想让您的脚本成为bash脚本,请替换
#!/bin/sh
Run Code Online (Sandbox Code Playgroud)
和:
#!/bin/bash
Run Code Online (Sandbox Code Playgroud)
然后当运行时./scriptname它会在bash. 如果您通过编写运行脚本,sh scriptname则必须bash scriptname改用。
如果您想消除大括号扩展,有几种选择。我建议seq使用命令替换,这可能是大括号扩展的最常见替代方法,它易于编写,并且很可能被其他人类读者理解。
代替{2..5},你可以写$(seq 2 5)。由于没有加引号-那就是,因为它是$( )不"$( )"-场分裂(这bash被称为分词)上的结果执行。只要您没有将控制字段拆分的IFSshell 变量设置为包含任何数字或不包含换行符的值,这将执行您想要的操作。
(通配符——也称为文件名扩展,也称为路径名扩展——也对不带引号的命令替换的结果执行,但 的输出seq将不包含通配符 ?, *, 或[,因此在这种情况下不起作用。)
请注意,seq是不规范的POSIX。这几乎适用于任何 GNU/Linux 系统和其他一些类 Unix 操作系统,但一些类 Unix 操作系统seq默认没有安装(他们通常会安装jot),因此不能保证在所有 Unix 上都可以使用 -比如操作系统。
当您在终端中运行脚本时,您使用的 shell 很可能是 Bash。Bash 支持大括号扩展,因此{2..5}可以扩展到序列,2 3 4 5并且您的脚本可以按预期工作。
但是,当您将相同的 scipt 添加到文件时,它不起作用。这是由于您使用的shebang ( #!/bin/sh),它告诉您的脚本在 sh shell 中运行,该 shell 符合 POSIX 并且不支持大括号扩展。
所以你有两个选择:
#!/bin/bash或另一个支持大括号扩展的shell。或者
#!/bin/sh并替换{2..5}为$(seq 2 5)。另请查看有关sh 和 Bash 之间差异的这个有趣的 Stack Overflow 问题。