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 / 问题 / 466496
Accepted
Sergiy Kolodyazhnyy
Sergiy Kolodyazhnyy
Asked: 2018-09-03 22:00:45 +0800 CST2018-09-03 22:00:45 +0800 CST 2018-09-03 22:00:45 +0800 CST

为什么在简单的 bash 命令中没有明显的克隆或分叉以及它是如何完成的?

  • 772

考虑以下(sh存在/bin/dash):

$ strace -e trace=process sh -c 'grep "^Pid:" /proc/self/status /proc/$$/status'
execve("/bin/sh", ["sh", "-c", "grep \"^Pid:\" /proc/self/status /"...], [/* 47 vars */]) = 0
arch_prctl(ARCH_SET_FS, 0x7fcc8b661540) = 0
clone(child_stack=NULL, flags=CLONE_CHILD_CLEARTID|CLONE_CHILD_SETTID|SIGCHLD, child_tidptr=0x7fcc8b661810) = 24865
wait4(-1, /proc/self/status:Pid:    24865
/proc/24864/status:Pid: 24864
[{WIFEXITED(s) && WEXITSTATUS(s) == 0}], 0, NULL) = 24865
--- SIGCHLD {si_signo=SIGCHLD, si_code=CLD_EXITED, si_pid=24865, si_uid=1000, si_status=0, si_utime=0, si_stime=0} ---
exit_group(0)                           = ?
+++ exited with 0 +++

没有什么不寻常的,从主 shell 进程中grep替换了一个分叉的进程(这里通过 完成)。clone()到目前为止,一切都很好。

现在使用 bash 4.4:

$ strace -e trace=process bash -c 'grep "^Pid:" /proc/self/status /proc/$$/status'
execve("/bin/bash", ["bash", "-c", "grep \"^Pid:\" /proc/self/status /"...], [/* 47 vars */]) = 0
arch_prctl(ARCH_SET_FS, 0x7f8416b88740) = 0
execve("/bin/grep", ["grep", "^Pid:", "/proc/self/status", "/proc/25798/status"], [/* 47 vars */]) = 0
arch_prctl(ARCH_SET_FS, 0x7f8113358b80) = 0
/proc/self/status:Pid:  25798
/proc/25798/status:Pid: 25798
exit_group(0)                           = ?
+++ exited with 0 +++

这里很明显的是grep假设 shell 进程的 pid 并且没有明显fork()或clone()调用。那么,问题是,如何在bash没有任何一个调用的情况下实现这样的杂技呢?

但是请注意,clone()如果命令包含 shell 重定向,则会出现系统调用,例如df > /dev/null

shell strace
  • 2 2 个回答
  • 2615 Views

2 个回答

  • Voted
  1. Best Answer
    Stéphane Chazelas
    2018-09-04T00:40:58+08:002018-09-04T00:40:58+08:00

    sh -c 'command line'通常用于诸如system("command line"), ssh host 'command line', vi's !,之类的东西cron,以及更普遍地用于解释命令行的任何东西,因此使其尽可能高效非常重要。

    分叉是昂贵的,在 CPU 时间、内存、分配的文件描述符方面......让一个 shell 进程在退出之前等待另一个进程只是浪费资源。此外,很难正确报告将执行命令的单独进程的退出状态(例如,当进程被终止时)。

    许多 shell 通常会尽量减少分叉的数量作为优化。即使是未优化的外壳也喜欢bash在sh -c cmdor(cmd in subshell)情况下这样做。与 ksh 或 zsh 不同,它不在bash -c 'cmd > redir'or 中执行此操作bash -c 'cmd1; cmd2'(在子shell 中相同)。ksh93 是在避免分叉方面走得最远的过程。

    在某些情况下无法进行优化,例如:

    sh < file
    

    哪里sh不能跳过最后一个命令的分叉,因为在该命令运行时可以将更多文本附加到脚本中。对于不可查找的文件,它无法检测到文件结尾,因为这可能意味着从文件中读取得太早太早。

    或者:

    sh -c 'trap "echo Ouch" INT; cmd'
    

    在执行“最后一个”命令后,shell 可能必须运行更多命令。

    • 10
  2. Sergiy Kolodyazhnyy
    2018-09-03T23:02:39+08:002018-09-03T23:02:39+08:00

    通过挖掘 bash 源代码,我发现如果没有管道或重定向,bash 实际上会忽略分叉。从execute_cmd.c 的第 1601 行开始:

      /* If this is a simple command, tell execute_disk_command that it
         might be able to get away without forking and simply exec.
         This means things like ( sleep 10 ) will only cause one fork.
         If we're timing the command or inverting its return value, however,
         we cannot do this optimization. */
      if ((user_subshell || user_coproc) && (tcom->type == cm_simple || tcom->type == cm_subshell) &&
          ((tcom->flags & CMD_TIME_PIPELINE) == 0) &&
          ((tcom->flags & CMD_INVERT_RETURN) == 0))
        {
          tcom->flags |= CMD_NO_FORK;
          if (tcom->type == cm_simple)
        tcom->value.Simple->flags |= CMD_NO_FORK;
        }
    

    稍后,这些标志开始execute_disk_command()起作用,它设置nofork整数变量,然后在尝试分叉之前对其进行检查。实际的命令本身将由分叉或父进程的execve()包装函数 shell_execve()运行,在这种情况下,它是实际的父进程。

    Stephane 的回答很好地解释了这种机制的原因。


    此问题范围之外的旁注:应注意,显然外壳是交互式的还是通过-c. 在执行命令之前会有一个分叉。从strace在交互式 shell ( strace -e trace=process -f -o test.trace bash) 上运行并检查输出文件可以看出这一点:

    19607 clone(child_stack=NULL, flags=CLONE_CHILD_CLEARTID|CLONE_CHILD_SETTID|SIGCHLD, child_t
    idptr=0x7f2d35e93a10) = 19628
    19607 wait4(-1,  <unfinished ...>
    19628 execve("/bin/true", ["/bin/true"], [/* 47 vars */]) = 0
    

    另请参阅为什么 bash 不会为简单命令生成子 shell?

    • 8

相关问题

  • 这个命令是如何工作的?mkfifo /tmp/f; 猫/tmp/f | /bin/sh -i 2>&1 | 数控 -l 1234 > /tmp/f

  • FreeBSD 的 sh:列出函数

  • 有没有办法让 ls 只显示某些目录的隐藏文件?

  • grep -v grep 有什么作用

  • 如何将带有〜的路径保存到变量中?

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