多次创建脚本到cp目录

noo*_*ntu 6 bash scripts

我有一个名为DS1. 我想4次复制,所以我也有DS2DS3DS4DS5

我可以从终端使用以下命令并且它有效:

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 之间的目录。不明白我做错了什么。

Eli*_*gan 8

{2..5}大括号扩展POSIX没有标准化大括号扩展。一些(但不是全部)广泛使用的Bourne 风格的shell 支持它。

您在 Ubuntu 上的终端中与之交互的 shell 是bash,除非您故意使用不同的 shell 。bash支持大括号扩展。但是家当你的脚本是sh,它在Ubuntu是一个符号链接dashdash不支持大括号扩展。

所以你可以:

  1. 使您的脚本成为bash脚本(或其他支持大括号扩展的 shell 的脚本,例如zshksh)。
  2. 将脚本中的大括号扩展替换为在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 上都可以使用 -比如操作系统。


Bea*_*nog 5

当您在终端中运行脚本时,您使用的 shell 很可能是 Bash。Bash 支持大括号扩展,因此{2..5}可以扩展到序列,2 3 4 5并且您的脚本可以按预期工作。

但是,当您将相同的 scipt 添加到文件时,它不起作用。这是由于您使用的shebang ( #!/bin/sh),它告诉您的脚本在 sh shell 中运行,该 shell 符合 POSIX 并且不支持大括号扩展。

所以你有两个选择:

  1. 将shebang 更改为#!/bin/bash或另一个支持大括号扩展的shell。

或者

  1. 使用#!/bin/sh并替换{2..5}$(seq 2 5)

另请查看有关sh 和 Bash 之间差异的这个有趣的 Stack Overflow 问题。