### Nohup命令
每个开发者都会躺过这个坑,在命令行跑一个后台程序,关闭终端后发现进程也退出了,网上搜一下发现要用`nohup`,究竟什么原因呢?
原来普通进程运行时默认会绑定TTY(虚拟终端),关闭终端后系统会给上面所有进程发送TERM信号,这时普通进程也就退出了。当然还有些进程不会退出,这就是后面将会提到的守护进程。
`Nohup`的原理也很简单,终端关闭后会给此终端下的每一个进程发送SIGHUP信号,而使用`nohup`运行的进程则会忽略这个信号,因此终端关闭后进程也不会退出。
### 举个例子
我们用Go实现最简单的Web服务器,代码web_server.go如下。
~~~
package main
import (
"fmt"
"net/http"
)
func handler(w http.ResponseWriter, r *http.Request) {
fmt.Println("Handle request")
}
func main() {
http.HandleFunc("/", handler)
http.ListenAndServe(":8000", nil)
}
~~~
然后在终端上运行,并测试一下。
~~~
➜ go build web_server.go
➜ ./web_server &
[1] 25967
➜ wget 127.0.0.1:8000
--2014-12-28 22:24:07-- http://127.0.0.1:8003/
Connecting to 127.0.0.1:8003... connected.
HTTP request sent, awaiting response... 200 OK
Length: 5 [text/plain]
Saving to: 'index.html.4'
100%[======================>] 5 --.-K/s in 0s
2014-12-28 22:24:07 (543 KB/s) - 'index.html' saved [5/5]
~~~
如果关闭终端,`curl`命令就连不上我们的Web服务器了。如果使用`nohup`运行呢?
~~~
➜ go build web_server.go
➜ nohup ./web_server &
[1] 25968
➜ exit
➜ wget 127.0.0.1:8003
--2014-12-28 22:24:11-- http://127.0.0.1:8003/
~~~
发现关闭终端对Web服务器进程没有任何影响,这正是我们预期的。这是运行守护进程最简单的方法,实际上标准的守护进程除了处理信号外,还要考虑这种因素,后面将会详述。
- 前言
- 致谢
- 概述
- 使用代码
- 使用Docker
- 进程基础
- 进程是什么
- Hello World
- PID
- PPID
- 使用PID
- 进程名字
- 进程参数
- 输入与输出
- 并发与并行
- 进程越多越好
- 进程状态
- 退出码
- 进程资源
- 死锁
- 活锁
- POSIX
- Nohup
- 运行进程
- Go编程实例
- 衍生新进程
- 执行外部程序
- 复制进程
- 进程进阶
- 文件锁
- 孤儿进程
- 僵尸进程
- 守护进程
- 进程间通信
- 信号
- Linux系统调用
- 文件描述符
- Epoll
- 共享内存
- Copy On Write
- Cgroups
- Namespaces
- 项目实例Run
- 项目架构
- 代码实现
- 注意事项
- 创建目录权限
- 捕获SIGKILL
- Sendfile系统调用
- 后记
- 参考书籍
- 项目学习
- 再次感谢