Skip to content
Permalink
Branch: master
Find file Copy path
Find file Copy path
Fetching contributors…
Cannot retrieve contributors at this time
96 lines (76 sloc) 2.18 KB

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

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

package main

import (
	"fmt"
	"os"
	"os/signal"
	"syscall"
	"time"
)

func handle(signal os.Signal) {
	fmt.Println("Received:", signal)
}

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

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

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

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

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

		for {
			sig := <-sigs
			switch sig {
			case os.Interrupt:
				handle(sig)
			case syscall.SIGTERM:
				handle(sig)
				os.Exit(0)
			case syscall.SIGUSR2:
				fmt.Println("Handling syscall.SIGUSR2!")
			default:
				fmt.Println("Ignoring:", sig)
			}
		}
	}()

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

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

handleAll.go剩余代码如下:

	for {
		fmt.Printf(".")
		time.Sleep(30 * time.Second)
	}
}

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

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

$ go build handleAll.go
$ ls -l handleAll
-rwxr-xr-x 1 mtsouk staff 2005216 Jan 18 08:25 handleAll
$ ./handleAll
.Ignoring: hangup
Handling syscall.SIGUSR2!
Ignoring: user defined signal 1
Received: interrupt
^CReceived: interrupt
Received: terminated

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

$ ps ax | grep ./handleAll | grep -v grep
49902 s003 S+ 0:00.00 ./handleAll
$ kill -s HUP 49902
$ kill -s USR2 49902
$ kill -s USR1 49902
$ kill -s INT 49902
$ kill -s TERM 49902
You can’t perform that action at this time.