处理所有信号

在本小节,你将学习如何处理所有信号,但只对真正感兴趣的信号作出响应。和上一节相比,它技术更好,并更安全。该技术代码参考handleAll.go,分为四部分。

handleAll.go第一部分代码如下:

  1. package main
  2. import (
  3. "fmt"
  4. "os"
  5. "os/signal"
  6. "syscall"
  7. "time"
  8. )
  9. func handle(signal os.Signal) {
  10. fmt.Println("Received:", signal)
  11. }

handleAll.go第二部分代码如下:

  1. func main() {
  2. sigs := make(chan os.Signal, 1)
  3. signal.Notify(sigs)

所以,所有的魔法都在signal.Notify(sigs)调用上。由于没有指定信号,所有输入信号都将被处理。

你可以在同一程序中使用不同的通道和相同的信号多次调用signal.Notify。在这种情况下,每个相关通道都将收到一份它要处理的信号副本!

handleAll.go第三部分代码如下:

  1. for {
  2. sig := <-sigs
  3. switch sig {
  4. case os.Interrupt:
  5. handle(sig)
  6. case syscall.SIGTERM:
  7. handle(sig)
  8. os.Exit(0)
  9. case syscall.SIGUSR2:
  10. fmt.Println("Handling syscall.SIGUSR2!")
  11. default:
  12. fmt.Println("Ignoring:", sig)
  13. }
  14. }
  15. }()

使用其中一个信号退出程序非常方便。

这给了你在需要时在程序中做一些内务管理的机会。此时,syscall.SIGERM信号被用于这个目的。但这并不妨碍你使用SIGKILL来终止程序。

handleAll.go剩余代码如下:

  1. for {
  2. fmt.Printf(".")
  3. time.Sleep(30 * time.Second)
  4. }
  5. }

你仍需要调用time.Sleep()以阻止程序立即退出。

同样,最好使用go build工具编译handleAll.go生成可执行文件。在新的终端中执行handleAll会产生如下的输出:

  1. $ go build handleAll.go
  2. $ ls -l handleAll
  3. -rwxr-xr-x 1 mtsouk staff 2005216 Jan 18 08:25 handleAll
  4. $ ./handleAll
  5. .Ignoring: hangup
  6. Handling syscall.SIGUSR2!
  7. Ignoring: user defined signal 1
  8. Received: interrupt
  9. ^CReceived: interrupt
  10. Received: terminated

另一个终端命令输出如下:

  1. $ ps ax | grep ./handleAll | grep -v grep
  2. 49902 s003 S+ 0:00.00 ./handleAll
  3. $ kill -s HUP 49902
  4. $ kill -s USR2 49902
  5. $ kill -s USR1 49902
  6. $ kill -s INT 49902
  7. $ kill -s TERM 49902