暗无天日

=============>DarkSun的个人博客

为什么 nohup 在 crontab 中不起作用

很多初学者在 crontab 中运行长时间任务时,习惯性地加上 nohup ,像这样:

* * * * * nohup /path/to/long-running-command &

他们期望命令的输出会被重定向到 nohup.out 文件中。但实际结果是: nohup.out 根本没有被创建,命令的输出却出现在了系统邮件中(可以用 mailx 命令查看)。

为什么会这样?要理解这个问题,我们需要先搞清楚两件事: nohup 到底做了什么,以及 crontab 是怎么执行命令的。

nohup 到底做了什么

在之前的文章 nohup,setsid与disown的不同之处 中,我详细分析了 nohup 的原理。这里简要回顾一下。

nohup 做两件事:

  1. 忽略 SIGHUP 信号(即 signal(SIGHUP, SIG_IGN) ),防止进程在终端关闭时被杀死
  2. 将输出重定向到 nohup.out 文件

但关键是第二点并不是无条件的。让我们看看 coreutils 中 nohup 的源码( src/nohup.c ):

bool ignoring_input = isatty (STDIN_FILENO);
bool redirecting_stdout = isatty (STDOUT_FILENO);
bool redirecting_stderr = isatty (STDERR_FILENO);

nohup 先用 isatty() 检查每个标准流是否连接到终端。只有当 stdout 是终端 的时候,才会将它重定向到 nohup.out

if (redirecting_stdout)
  {
    /* 尝试打开 nohup.out,失败则尝试 $HOME/nohup.out */
    fd_reopen (STDOUT_FILENO, file, flags, mode);
  }

也就是说,如果你的 stdout 已经被重定向到了文件或管道, nohup 会认为"输出已经有地方去了",就不会再创建 nohup.out 了。 stderr 也一样: nohup 同样会检查 isatty(STDERR_FILENO) ,只有 stderr 是终端时才会将它重定向到 nohup.out

这是理解问题的关键。

crontab 的运行环境

要理解为什么 nohup 在 crontab 中不生效,我们需要看看 cron 是怎么执行用户命令的。

当你编辑 crontab 添加一条任务后, cron 守护进程会在指定时间执行你的命令。关键在于 cron 是怎么执行这些命令的:

  1. cron 会 fork 出一个子进程
  2. 这个子进程会调用 exec("/bin/sh", "-c", "你的命令") 来执行任务
  3. 在执行之前,cron 会将子进程的 stdout 和 stderr 连接到 管道 ,而不是终端

这意味着什么?意味着在你的命令看来:

  • stdin 是 /dev/null (不是终端)
  • stdout 是一个管道(不是终端)
  • stderr 是一个管道(不是终端)
  • 没有控制终端(controlling terminal)
  • 工作目录默认是 HOME

你可以用一个简单的方法验证:在 crontab 中执行 tty 命令,它会告诉你当前的终端设备:

* * * * * tty

你会收到一封系统邮件,内容类似 not a tty ,这就证明了 crontab 的执行环境中没有终端。

为什么 nohup 的重定向不生效

现在把前面的线索串起来:

  1. crontab 执行命令时,stdout 和 stderr 是 管道 ,不是终端
  2. nohup 检查 isatty(STDOUT_FILENO) ,因为 stdout 是管道,返回 false
  3. nohup 认为"输出已经有地方去了",跳过重定向步骤
  4. 命令的输出顺着管道传给了 cron
  5. cron 将收到的输出发送到系统邮件

所以 nohup.out 根本就不会被创建!输出去了哪里?去了 cron 的邮件队列。

同时, nohup 的另一个功能——忽略 SIGHUP 信号——在 crontab 中也毫无意义。因为 cron 的执行环境根本没有终端会话,不存在"终端挂断"的场景,自然也不会有 SIGHUP 信号。

总结一下: nohup 在 crontab 中是 双重无用 的:

功能 在 crontab 中 原因
忽略 SIGHUP 不需要 cron 无终端会话,无 SIGHUP
输出重定向到 nohup.out 不触发 stdout 不是终端,isatty 返回 false

正确的做法

既然 nohup 在 crontab 中没用,那输出应该怎么处理?答案很简单:自己重定向。

* * * * * /path/to/command >> /path/to/output.log 2>&1

这样写就足够了,原因:

  • 不需要 nohup :cron 没有终端会话,不存在 SIGHUP 的问题
  • 不需要 & :cron 会为每个任务 fork 一个独立的子进程来执行,不会阻塞主调度循环,所以没必要手动放到后台。更何况使用 = &= 会让命令脱离 shell 成为孤儿进程,当 cron 关闭管道的读端后,命令尝试写入时会收到 SIGPIPE 信号,反而可能崩溃
  • 自己控制输出位置 :用 shell 的重定向语法精确指定输出文件

如果你想让命令在 cron 触发后持续运行(比如一个需要跑好几个小时的任务),直接执行就行。cron 会等待命令完成,不会中途杀掉进程。但要注意:如果命令运行时间超过了 crontab 的调度间隔,可能会出现重复执行的问题。这时候可以考虑在命令中加入锁机制(比如 flock ),或者使用 systemd timer 替代 crontab。

如果你想丢弃输出,也不想收到邮件:

* * * * * /path/to/command > /dev/null 2>&1

这比 nohup command > /dev/null 2>&1 & 简洁得多,而且效果完全一样。

linux和它的小伙伴