AskOverflow.Dev

AskOverflow.Dev Logo AskOverflow.Dev Logo

AskOverflow.Dev Navigation

  • 主页
  • 系统&网络
  • Ubuntu
  • Unix
  • DBA
  • Computer
  • Coding
  • LangChain

Mobile menu

Close
  • 主页
  • 系统&网络
    • 最新
    • 热门
    • 标签
  • Ubuntu
    • 最新
    • 热门
    • 标签
  • Unix
    • 最新
    • 标签
  • DBA
    • 最新
    • 标签
  • Computer
    • 最新
    • 标签
  • Coding
    • 最新
    • 标签
主页 / unix / 问题 / 490000
Accepted
Jeremy
Jeremy
Asked: 2018-12-20 12:04:07 +0800 CST2018-12-20 12:04:07 +0800 CST 2018-12-20 12:04:07 +0800 CST

针对匹配大括号 + glob 模式的文件运行多个命令而不重复它

  • 772

我想对一组由大括号和全局模式匹配的文件运行一系列命令,而不是在整个地方复制粘贴模式。我一直试图通过将模式放入变量中来做到这一点,但一直无法弄清楚如何使该变量像原始模式一样工作。我怎么能这样做,或者以其他方式解决这个问题?

例如,如何针对与标量变量中定义cat的模式匹配的文件运行?src/component\ {a,b,c}/*.ccomponent_source_code

示例上下文和复制

set -euo pipefail;
mkdir "src/" "dist/";
trap 'rm -r "src/" "dist/"' EXIT;

我有一个项目,其结构类似于以下内容(尽管内容更有用)。

>"src/README.md"                 date;
mkdir "src/component a/";
>"src/component a/program.c"     date;
>"src/component a/tests.c"       date;
>"src/component a/budget\$.txt"  date;
mkdir "src/component b/";
>"src/component b/program.c"     date;
>"src/component b/tests.c"       date;
>"src/component b/braces{}.txt"  date;
mkdir "src/component c/";
>"src/component c/program.c"     date;
>"src/component c/tests.c"       date;
>"src/component c/test data.txt" date;
mkdir "src/docs";
>"src/docs/test data.txt"        date;

我有需要针对多个组件中的相关文件的构建步骤。我已经用大括号 + glob 模式定义了变量来匹配这些文件集。

readonly component_paths_pattern="src/component\ {a,b,c}";
readonly component_data_pattern="${component_paths_pattern}/*.txt";
readonly component_code_pattern="${component_paths_pattern}/*.c";

当我手动将这些模式复制到示例命令中时,它们与预期的文件匹配。

>"dist/support.txt" cat src/component\ {a,b,c}/*.txt;
test -s "dist/all test data.txt";

>"dist/all.c" cat src/component\ {a,b,c}/*.c;
test -s "dist/all.c";

如果我只需要引用一次就可以了,但实际上我需要从构建脚本的不同部分多次引用相同的文件集,因此我希望重用变量中的模式。但是,我一直无法弄清楚如何使其正常工作。

set -x;

尝试失败的解决方案

不带引号的变量扩展(拆分+通配符)

>"dist/support.txt" cat ${component_data_pattern};

我认为这失败了,因为模式包含一个空格,所以它被分成两个单独的 glob 模式参数,它们自己都不匹配任何东西。

+ cat 'src/component\' '{a,b,c}/*.txt'
cat: src/component\: No such file or directory
cat: {a,b,c}/*.txt: No such file or directory

引用变量展开

>"dist/support.txt" cat "${component_data_pattern}";

我认为这失败了,因为大括号扩展发生在变量扩展之前,所以大括号没有机会在这里扩展。

+ cat 'src/component\ {a,b,c}/*.txt'
cat: src/component\ {a,b,c}/*.txt: No such file or directory

参数列表中的 Eval 和 Echo

>"dist/support.txt" cat $(eval "echo ${component_data_pattern}");

如果不引用子命令扩展,我认为这会失败,因为某些生成的路径包含空格,导致它们被拆分为单独的参数。

++ eval 'echo src/component\ {a,b,c}/*.txt'
+++ echo 'src/component a/budget$.txt' 'src/component b/braces{}.txt' 'src/component c/test data.txt'
+ cat src/component 'a/budget$.txt' src/component 'b/braces{}.txt' src/component c/test data.txt
cat: src/component: No such file or directory
[...]
>"dist/support.txt" cat "$(eval "echo ${component_data_pattern}")";

如果我确实引用了子命令扩展,我认为它会失败,因为这会将所有路径连接成一个字符串,从而产生一个长的无效路径。

++ eval 'echo src/component\ {a,b,c}/*.txt'
+++ echo 'src/component a/budget$.txt' 'src/component b/braces{}.txt' 'src/component c/test data.txt'
+ cat 'src/component a/budget$.txt src/component b/braces{}.txt src/component c/test data.txt'
cat: src/component a/budget$.txt src/component b/braces{}.txt src/component c/test data.txt: No such file or directory

参数列表中的 Eval 和 Printf %q

由于类似的原因,使用printf '%q '而不是失败。echo

>"dist/support.txt" cat "$(eval "printf '%q ' ${component_data_pattern}")";
++ eval 'printf '\''%q '\'' src/component\ {a,b,c}/*.txt'
+++ printf '%q ' 'src/component a/budget$.txt' 'src/component b/braces{}.txt' 'src/component c/test data.txt'
+ cat 'src/component\ a/budget\$.txt src/component\ b/braces\{\}.txt src/component\ c/test\ data.txt '
cat: src/component\ a/budget\$.txt src/component\ b/braces\{\}.txt src/component\ c/test\ data.txt : No such file or directory
>"dist/support.txt" cat $(eval "printf '%q ' ${component_data_pattern}");
++ eval 'printf '\''%q '\'' src/component\ {a,b,c}/*.txt'
+++ printf '%q ' 'src/component a/budget$.txt' 'src/component b/braces{}.txt' 'src/component c/test data.txt'
+ cat 'src/component\' 'a/budget\$.txt' 'src/component\' 'b/braces\{\}.txt' 'src/component\' 'c/test\' data.txt
cat: src/component\: No such file or directory
[...]
bash wildcards
  • 2 2 个回答
  • 368 Views

2 个回答

  • Voted
  1. Jeremy
    2018-12-20T12:04:17+08:002018-12-20T12:04:17+08:00

    评估整个命令(不仅仅是参数)

    eval ">\"dist/support.txt\" cat ${component_data_pattern}";
    test -s "dist/all.c";
    

    我不喜欢这个,但它有效。鉴于我们正在尝试扩展一个包含大括号和文件 glob 的模式,其中一个发生在变量扩展之前,另一个发生在变量扩展之后,可能没有其他选择可以这样:手动将变量扩展为包含整个命令调用的字符串,并将该字符串用作evalor的参数bash -c。不要忘记使用 . 转义任何内部引号\"。

    在上面的示例中,没有其他参数。如果还有其他参数并且它们也使用某种替换,则您需要转义这些参数(使用\$、\*、\{或\}),以便在最终评估命令并且可以在上下文中解释它们之前不会扩展它们。

    readonly annoying_arg="$PWD/src/docs/test data.txt";
    eval ">\"dist/support.txt\" cat ${component_data_pattern} \"\$annoying_arg\"";
    test -s "dist/all.c";
    
    • 1
  2. Best Answer
    Kusalananda
    2018-12-20T12:37:21+08:002018-12-20T12:37:21+08:00

    使用数组,不要将文件名通配模式存储在变量中(让它们扩展为匹配的路径名):

    component_dirs=( 'src/component '{a,b,c} )
    
    component_data=()
    component_code=()
    
    for dir in "${component_dirs[@]}"; do
        component_data+=( "$dir"/*.txt )
        component_code+=( "$dir"/*.c   )
    done
    

    然后你可以做,例如,

    cat "${component_data[@]}"
    

    除非该数组包含数百或数千个路径名。

    • 1

相关问题

  • 通过命令的标准输出以编程方式导出环境变量[重复]

  • 从文本文件传递变量的奇怪问题

  • 虽然行读取保持转义空间?

  • `tee` 和 `bash` 进程替换顺序

  • 运行一个非常慢的脚本直到它成功

Sidebar

Stats

  • 问题 205573
  • 回答 270741
  • 最佳答案 135370
  • 用户 68524
  • 热门
  • 回答
  • Marko Smith

    如何将 GPG 私钥和公钥导出到文件

    • 4 个回答
  • Marko Smith

    ssh 无法协商:“找不到匹配的密码”,正在拒绝 cbc

    • 4 个回答
  • Marko Smith

    我们如何运行存储在变量中的命令?

    • 5 个回答
  • Marko Smith

    如何配置 systemd-resolved 和 systemd-networkd 以使用本地 DNS 服务器来解析本地域和远程 DNS 服务器来解析远程域?

    • 3 个回答
  • Marko Smith

    如何卸载内核模块“nvidia-drm”?

    • 13 个回答
  • Marko Smith

    dist-upgrade 后 Kali Linux 中的 apt-get update 错误 [重复]

    • 2 个回答
  • Marko Smith

    如何从 systemctl 服务日志中查看最新的 x 行

    • 5 个回答
  • Marko Smith

    Nano - 跳转到文件末尾

    • 8 个回答
  • Marko Smith

    grub 错误:你需要先加载内核

    • 4 个回答
  • Marko Smith

    如何下载软件包而不是使用 apt-get 命令安装它?

    • 7 个回答
  • Martin Hope
    rocky 如何将 GPG 私钥和公钥导出到文件 2018-11-16 05:36:15 +0800 CST
  • Martin Hope
    Wong Jia Hau ssh-add 返回:“连接代理时出错:没有这样的文件或目录” 2018-08-24 23:28:13 +0800 CST
  • Martin Hope
    Evan Carroll systemctl 状态显示:“状态:降级” 2018-06-03 18:48:17 +0800 CST
  • Martin Hope
    Tim 我们如何运行存储在变量中的命令? 2018-05-21 04:46:29 +0800 CST
  • Martin Hope
    Ankur S 为什么 /dev/null 是一个文件?为什么它的功能不作为一个简单的程序来实现? 2018-04-17 07:28:04 +0800 CST
  • Martin Hope
    user3191334 如何从 systemctl 服务日志中查看最新的 x 行 2018-02-07 00:14:16 +0800 CST
  • Martin Hope
    Marko Pacak Nano - 跳转到文件末尾 2018-02-01 01:53:03 +0800 CST
  • Martin Hope
    Kidburla 为什么真假这么大? 2018-01-26 12:14:47 +0800 CST
  • Martin Hope
    Christos Baziotis 在一个巨大的(70GB)、一行、文本文件中替换字符串 2017-12-30 06:58:33 +0800 CST
  • Martin Hope
    Bagas Sanjaya 为什么 Linux 使用 LF 作为换行符? 2017-12-20 05:48:21 +0800 CST

热门标签

linux bash debian shell-script text-processing ubuntu centos shell awk ssh

Explore

  • 主页
  • 问题
    • 最新
    • 热门
  • 标签
  • 帮助

Footer

AskOverflow.Dev

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve