对于这个函数, 一般可能用的比较少, 这里稍微介绍下. 它有 3 个参数:
- args[0]: 可执行文件的路径(相对路径, 绝对路径或者 PATH 中的路径都可以)
- args[1]: 命令的参数
- args[2]: 命令的执行的环境变量, os.Environ() 表示继承 caller 的环境变量
当 syscall.Exec 执行时, 在它之前的所有未执行完的程序都会被中止(包括在 go routine 中执行的程序),
然后执行 syscall.Exec 调用的命令, 该命令还保持在之前程序的 PID 下执行.
syscall.Exec 是最后一条执行的代码, 重启时在它之后可以有代码, 但是都不会被执行到, 包括 defer 中的代码.
下面是个小例子(通过这个例子可以验证上面的结论):
1 package main
2
3 import (
4 "fmt"
5 "log"
6 "os"
7 "syscall"
8 "time"
9
10 "github.com/fsnotify/fsnotify"
11 "github.com/kardianos/osext"
12 )
13
14 func syscallExec() {
15 watcher, err := setupWatcher()
16 if err != nil {
17 log.Fatal(err)
18 }
19 defer finally(watcher)
20
21 fmt.Printf("current pid: %dn", os.Getpid())
22 var count = 0
23
24 go func(count int) {
25 for {
26 fmt.Printf(">>> count in GO ROUTINE: %dn", count)
27 count++
28 time.Sleep(1 * time.Second)
29 }
30 }(count)
31
32 for {
33 fmt.Printf(">>> count in MAIN: %dn", count)
34 count++
35 time.Sleep(1 * time.Second)
36 }
37 }
38
39 func finally(watcher chan struct{}) {
40 // 重启时没有执行此函数
41 fmt.Println("exit original exec")
42 close(watcher)
43 }
44
45 func setupWatcher() (chan struct{}, error) {
46 file, err := osext.Executable()
47 if err != nil {
48 return nil, err
49 }
50 log.Printf("watching %qn", file)
51 w, err := fsnotify.NewWatcher()
52 if err != nil {
53 return nil, err
54 }
55 done := make(chan struct{})
56 go func() {
57 select {
58 case e := <-w.Events:
59 log.Printf("watcher received: %v", e)
60 err := syscall.Exec(file, os.Args, os.Environ())
61 if err != nil {
62 log.Fatal(err)
63 }
64 case err := <-w.Errors:
65 log.Printf("watcher error: %+v", err)
66 case <-done:
67 log.Print("watcher shutting down")
68 return
69 }
70 }()
71 err = w.Add(file)
72 if err != nil {
73 return nil, err
74 }
75 return done, nil
76 }
内容来源于网络如有侵权请私信删除