我正在尝试使用以下代码省略最后一个文件,同时将其余文件复制到目标文件夹中:
ls -Art | tail -n 1 | xargs -I% cp !(%) ~/test_dst/folder1
前两个管道获取最新文件的名称,并将其通过管道传输到 xargs 中,然后将其添加到 cp 命令中,我使用 extglob 的 ! 通配符复制其他所有内容。问题是,这会复制每个文件!我试图通过确定此打印的第一部分(set -x 已打开)来调试它:
> ~/test_src/folder1$ ls -Art | tail -n 1
+ tail -n 1
+ ls --color=auto -Art
file4
成功!这是实际正确的文件!然后我尝试使用通配符打印它:
> ~/test_src/folder1$ ls -Art | tail -n 1 | xargs -I% echo !(%)
+ tail -n 1
+ ls --color=auto -Art
+ xargs -I% echo file1 file2 file3 file4
file1 file2 file3 file4
它包括省略的文件??我试图自己对命令进行硬编码,你知道吗,它实际上是正确的:
> ~/test_src/folder1$ echo !(file4)
+ echo file1 file2 file3
file1 file2 file3
我究竟做错了什么?这基本上是我第一次尝试 shell 脚本,它让我发疯,非常感谢任何提示。我更愿意调试这个特定的方法(这正是我想到的方式),但我绝对愿意接受任何建议或替代方法。谢谢你!
是的,这行不通。您尝试组合的两个功能的解释顺序错误。
Extglob 是 shell(命令解释器)的一个特性,正如您在“trace”输出中所注意到的,它们在'xargs' 命令实际运行之前由 bash 扩展。
如果 extglobs——或者任何 globs,真的——按原样传递给 xargs(或“echo”或“cp”),它不知道如何处理它们;例如,'cp' 会认为 "!(%)" 或 "!(file1)" 就是要复制的文件名。
另一方面,xargs不是shell 的一部分——它和其他程序一样是一个独立的程序,而 bash 不知道“%”的含义,也不知道您正在向 xargs 传递另一个命令。
因此,当
!(%)
扩展 extglob 时,它只为整个命令扩展一次%
——不是为每个 xargs 输入扩展一次,而是只为Bash 在命令中看到的文字扩展一次。这当然会导致所有未命名的文件%
。一种方法是用 shell 自己的扩展替换 xargs - 与 xargs 不同,shell 变量在 extglobs之前扩展,所以如果你在变量中有文件名......
...您可以在 extglob 中使用它:
(两者可以组合成
cp !("$(ls...)") ~/...
。)也可以做相反的事情并保留 xargs,但删除 extglob。除了获取最后一个文件
tail
然后需要额外的步骤来获取其他所有内容之外,您还可以直接获取除最后一个文件之外的所有内容head
('head' 和 'tail' 的负参数表示“除了最后一个 N”):或者,让 xargs
cp
为所有文件一次调用一个,假设您的系统具有 GNU Coreutilscp -t <target>
选项:旁注:Linux 上的文件名可以包含换行符。上面的许多示例都假设你没有这样的文件名,因为你真的不应该——但在编写要部署在未知系统上的脚本时,这是需要考虑的事情,“防御性”编程通常会涉及到
find ... -print0
和xargs -0
使用以空值分隔的列表,而不是换行符分隔的列表。(Shell 通配符通常可以很好地处理它。)