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 / 问题 / 493808
Accepted
joker
joker
Asked: 2019-01-11 12:04:59 +0800 CST2019-01-11 12:04:59 +0800 CST 2019-01-11 12:04:59 +0800 CST

使用引号的命令差异(查找)

  • 772

我正在阅读手册页,find发现自己对以下命令感到困惑。一个和它对应的一个有什么区别。

  1. 以下两个命令有什么区别:

    find -execdir command "{}" \;
    find -execdir "command {}" \;
    

混淆的原因:我认为引用应该指示外壳将引用的部分作为一个块。因此,当我看到第二个时,我认为它会失败,因为没有命令command <file-name>。

  1. 以下两者有什么区别:

    find -execdir bash -c "command" "{}" \;
    find -execdir bash -c "command {}" \;
    

混淆的原因:根据我的理解,在第二个版本中将命令与花括号一起括起来应该作为一个整体传递给 bash 命令,并且find不应该用相应的文件名解释大括号。

  1. 以下两者有什么区别:

    find -execdir bash -c "something \"$@\"" {} \;
    find -execdir bash -c 'something "$@"' bash {} \;
    

据我了解,两者是相同的。将大括号传递给 shell,第二个版本,如何比第一个更安全。

更新

刚刚注意到问题#3 的第一个版本的命令不起作用!尝试了以下(没有工作):

find -execdir bash -c 'something \"$@\"' {} \; # changed external quotes to single ones.
find -execdir bash -c "something $@" {} \; # removed the inner quotes.
find -execdir bash -c 'something $@' {} \; # same as above but with outer single quotes instead.

我在这里想念什么?我相信我应该能够将大括号从包含命令的引号中去掉?

find command-line
  • 1 1 个回答
  • 333 Views

1 个回答

  • Voted
  1. Best Answer
    Gordon Davisson
    2019-01-11T16:13:06+08:002019-01-11T16:13:06+08:00

    这(正如您所注意到的)相当复杂。我会试着解释一下。根据命令经过的解析/处理顺序进行思考,并观察每一步发生的情况是有帮助的。通常,该过程如下所示:

    1. shell 解析命令行,将其分解为标记(“单词”),替换变量引用等,并删除引号和转义符(在应用它们的效果之后)。然后它(通常)运行第一个“单词”作为命令名称(在这些情况下为“查找”),并将其余单词作为参数传递给它。
    2. find搜索文件,并将其“ -execdir”和“ ;”之间的内容作为命令运行。请注意,它将“ {}”替换为匹配的文件名,但不进行其他解析——它只是运行“ -execdir”之后的第一个 arg 作为命令名,并将以下参数作为其参数传递给它。
    3. 如果该命令恰好是bash并且它被传递了-c选项,它会立即将参数解析-c为命令字符串(有点像微型 shell 脚本),并将其其余参数作为该迷你脚本的参数。

    好的,在我深入研究之前还有一些注意事项:我正在使用 BSD find,它要求明确指定要搜索的目录,所以我将使用find . -execdir ...而不是find -execdir .... 我在一个包含文件“foo.txt”和“z@$%^;*;echo wheee.jpg”的目录中(以说明使用bash -c错误的风险)。最后,我pargs在我的二进制目录中调用了一个简短的脚本,它打印了它的参数(或者如果没有得到任何参数就会抱怨)。

    问题一:

    现在让我们试试第一个问题中的两个命令:

    $ find . -execdir pargs "{}" \;
    pargs got 1 argument(s): '.'
    pargs got 1 argument(s): 'foo.txt'
    pargs got 1 argument(s): 'z@$%^;*;echo wheee.jpg'
    $ find . -execdir "pargs {}" \;
    find: pargs .: No such file or directory
    find: pargs foo.txt: No such file or directory
    find: pargs z@$%^;*;echo wheee.jpg: No such file or directory
    

    这符合您的期望:第一个有效(顺便说一句,您可以省略双引号{}),第二个失败,因为空格和文件名被视为命令名称的一部分,而不是它的参数。

    顺便说一句,也可以使用-exec[dir] ... +而不是-exec[dir] \;-- 这告诉find尽可能少地运行命令,并一次传递一堆文件名:

    $ find . -execdir pargs {} +
    pargs got 3 argument(s): '.' 'foo.txt' 'z@$%^;*;echo wheee.jpg'
    

    问题二:

    这次我将一次选择一个选项:

    $ find . -execdir bash -c "pargs" "{}" \;
    pargs didn't get any arguments
    pargs didn't get any arguments
    pargs didn't get any arguments
    

    “嗯”,你说?这里发生的是使用“ ”、“ ”、“ ”之bash类的参数列表运行。该选项告诉运行它的下一个参数(“pargs”),就像一个微型 shell 脚本,像这样:-cpargsfoo.txt-cbash

    #!/bin/bash
    pargs
    

    ...然后将“迷你脚本”传递给参数“foo.txt”(稍后会详细介绍)。但是那个迷你脚本并没有对它的参数做任何事情——具体来说,它不会将它们传递给pargs命令,所以pargs什么都看不到。(我将在第三个问题中找到正确的方法。)现在,让我们尝试第二个问题的第二个替代方案:

    $ find . -execdir bash -c "pargs {}" \;
    pargs got 1 argument(s): '.'
    pargs got 1 argument(s): 'foo.txt'
    pargs got 1 argument(s): 'z@$%^'
    bash: foo.txt: command not found
    wheee.jpg
    

    现在一切都在起作用,但只是在某种程度上。bash使用参数“ -c”和“pargs”+文件名运行,这与“。”的预期相同。和“foo.txt”,但是当你传递bash参数“ -c”和“ pargs z@$%^;*;echo wheee.jpg”时,它现在运行相当于它的迷你脚本:

    #!/bin/bash
    pargs z@$%^;*;echo wheee.jpg
    

    因此 bash 会将其拆分为三个用分号分隔的命令:

    1. “ pargs z@$%^”(你看到的效果)
    2. " *",扩展为单词 " foo.txt" 和 " z@$%^;*;echo wheee.jpg",因此尝试foo.txt作为命令运行并将另一个文件名作为参数传递给它。没有该名称的命令,因此它给出了适当的错误。
    3. “ echo echo wheee.jpg”,这是一个非常合理的命令,正如你所见,它会在终端上打印“wheee.jpg”。

    因此它适用于具有纯名称的文件,但是当它遇到包含 shell 语法的文件名时,它开始尝试执行部分文件名。这就是为什么这种做事方式不被认为是安全的。

    问题三:

    同样,我将一次查看一个选项:

    $ find . -execdir bash -c "pargs \"$@\"" {} \;
    pargs got 1 argument(s): ''
    pargs got 1 argument(s): ''
    pargs got 1 argument(s): ''
    $ 
    

    再一次,我听到你说“嗯????” 这里最大的问题是它$@没有转义或单引号,因此它在传递给find. 我将在这里pargs展示find实际得到的参数:

    $ pargs . -execdir bash -c "pargs \"$@\"" {} \;
    pargs got 7 argument(s): '.' '-execdir' 'bash' '-c' 'pargs ""' '{}' ';'
    

    请注意,$@刚刚消失了,因为我在一个没有收到任何参数(或使用set命令设置它们)的交互式 shell 中运行它。因此,我们正在运行这个迷你脚本:

    #!/bin/bash
    pargs ""
    

    ...这解释了为什么pargs得到一个空参数。

    如果这是在已接收参数的脚本中,事情会更加混乱。转义(或单引号)$解决了这个问题,但仍然不太有效:

    $ find . -execdir bash -c 'pargs "$@"' {} \;
    pargs didn't get any arguments
    pargs didn't get any arguments
    pargs didn't get any arguments
    

    这里的问题是bash将迷你脚本之后的下一个参数视为迷你脚本的名称(迷你脚本可用作$0,但不包含在 中$@),而不是常规参数(即$1)。这是一个演示这个的常规脚本:

    $ cat argdemo.sh 
    #!/bin/bash
    echo "My name is $0; I received these arguments: $@"
    $ ./argdemo.sh foo bar baz
    My name is ./argdemo.sh; I received these arguments: foo bar baz
    

    现在用一个类似bash -c的小脚本试试这个:

    $ bash -c 'echo "My name is $0; I received these arguments: $@"' foo bar baz
    My name is foo; I received these arguments: bar baz
    

    解决这个问题的标准方法是添加一个虚拟脚本名称参数(如“bash”),以便实际参数以通常的方式显示:

    $ bash -c 'echo "My name is $0; I received these arguments: $@"' mini-script foo bar baz
    My name is mini-script; I received these arguments: foo bar baz
    

    这正是您的第二个选项所做的,将“bash”作为脚本名称传递,找到的文件名传递为$1:

    $ find . -execdir bash -c 'pargs "$@"' bash {} \;
    pargs got 1 argument(s): '.'
    pargs got 1 argument(s): 'foo.txt'
    pargs got 1 argument(s): 'z@$%^;*;echo wheee.jpg'
    

    这终于奏效了——真的,即使是奇怪的文件名。这就是为什么这(或您的第一个问题中的第一个选项)被认为是使用find -exec[dir]. 您也可以将其与以下-exec[dir] ... +方法一起使用:

    $ find . -execdir bash -c 'pargs "$@"' bash {} +
    pargs got 3 argument(s): '.' 'foo.txt' 'z@$%^;*;echo wheee.jpg'
    
    • 3

相关问题

  • 需要许多参数的实用程序的推荐界面是什么?[关闭]

  • 远程运行 X 应用程序,在远程主机上运行 GUI [关闭]

  • 使 mysql CLI 以交互方式向我询问密码

  • GNU find:在-exec中获取绝对和相对路径

  • 没有服务器的命令行 pub/sub?

Sidebar

Stats

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

    模块 i915 可能缺少固件 /lib/firmware/i915/*

    • 3 个回答
  • Marko Smith

    无法获取 jessie backports 存储库

    • 4 个回答
  • Marko Smith

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

    • 4 个回答
  • Marko Smith

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

    • 5 个回答
  • Marko Smith

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

    • 3 个回答
  • 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
    user12345 无法获取 jessie backports 存储库 2019-03-27 04:39:28 +0800 CST
  • Martin Hope
    Carl 为什么大多数 systemd 示例都包含 WantedBy=multi-user.target? 2019-03-15 11:49:25 +0800 CST
  • Martin Hope
    rocky 如何将 GPG 私钥和公钥导出到文件 2018-11-16 05:36:15 +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

热门标签

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