Go语言daemon启动的解决方法.linux平台

1、使用nohup ./commond & 这种方法最简单.
nohup这个命令可以把程序放后台运行,顺便通过1>和2>把标准输出和标准错误重定向到文件,这样程序崩溃时才会有记录可查,这两者和程序的日志最好是分开,混在一起没办法判断轻重缓急:
nohup ./server 1> server.out 2> server.err
进程启动时候记录下自己的pid:

if pid := syscall.Getpid(); pid != 1 {
    ioutil.WriteFile("server.pid", []byte(strconv.Itoa(pid)), 0777)
    defer os.Remove("server.pid")
}

同时监听系统发来的kill信号,在收到kill信号时做些收尾工作:

signal.Notify(sigTERM, syscall.SIGTERM)

这样就可以用kill命令关闭服务进程了:

kill `cat game_server.pid`

2、这种方法可以使用信号.

package main

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

func main() {
    if os.Getppid() != 1 {
        args := append([]string{os.Args[0]}, os.Args[1:]...)
        os.StartProcess(os.Args[0], args, &os.ProcAttr{Files: []*os.File{os.Stdin, os.Stdout, os.Stderr}})
        return
    }
    go func() {
        var c chan os.Signal = make(chan os.Signal, 1)
        go signal.Notify(c, os.Kill, os.Interrupt, syscall.SIGHUP, syscall.SIGUSR2)
        File, err := os.Create("/tmp/收到的信号.")
        defer File.Close()
        for {
            s := <-c
            File.WriteString(s.String())
        }
    }()
    F, err := os.Create("/tmp/log")
    if err != nil {
        fmt.Println(err)
        return
    }
    defer F.Close()
    for {
        fmt.Fprintln(F, "Hello World")
        time.Sleep(3e9)
    }
}

3、这种方法比较底层,但是不能使用信号

package main

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

func daemon(nochdir, noclose int) int {
    var ret, ret2 uintptr
    var err syscall.Errno
    darwin := runtime.GOOS == "darwin"
    // already a daemon
    if syscall.Getppid() == 1 {
        return 0
    }
    // fork off the parent process
    ret, ret2, err = syscall.RawSyscall(syscall.SYS_FORK, 0, 0, 0)
    if err != 0 {
        return -1
    }
    // failure
    if ret2 < 0 {
        os.Exit(-1)
    }
    // handle exception for darwin
    if darwin && ret2 == 1 {
        ret = 0
    }
    // if we got a good PID, then we call exit the parent process.
    if ret > 0 {
        os.Exit(0)
    }
    /* Change the file mode mask */
    _ = syscall.Umask(0)

    // create a new SID for the child process
    s_ret, s_errno := syscall.Setsid()
    if s_errno != nil {
        log.Printf("Error: syscall.Setsid errno: %d", s_errno)
    }
    if s_ret < 0 {
        return -1
    }
    if nochdir == 0 {
        os.Chdir("/")
    }
    if noclose == 0 {
        f, e := os.OpenFile("/dev/null", os.O_RDWR, 0)
        if e == nil {
            fd := f.Fd()
            syscall.Dup2(int(fd), int(os.Stdin.Fd()))
            syscall.Dup2(int(fd), int(os.Stdout.Fd()))
            syscall.Dup2(int(fd), int(os.Stderr.Fd()))
        }
    }
    return 0
}
func main() {
    daemon(0, 1)
    for {
        fmt.Println("hello")
        time.Sleep(1 * time.Second)
    }
}

原文地址:https://www.cnblogs.com/enumx/p/12316240.html

时间: 2024-11-09 20:07:38

Go语言daemon启动的解决方法.linux平台的相关文章

安装Wamp后 Apache无法启动的解决方法

安装Wamp后 Apache无法启动的解决方法,网上的解决方案可以说是五花八门,有些说了一大推,一点作用都起不到. 其实解决方法只需两步: 1.安装路径不能包含有中文,这个我不知道为什么,总之如果安装路径中包含有中文,接下来无论做多少配置工作都是徒劳. 2.安装好后 修改httpd.conf文件 把里面的80端口 改成其他端口 比方改为:8080 ,如果没有端口冲突可以不用修改. 做好以上这两步重启Wamp就可以了. 就这么简单,没必要搞太多东西.

应用程序无法正常启动0xc000007b解决方法

一 问题 1 应用程序无法正常启动0xc000007b解决方法 2 无法启动此程序,因为计算机中丢失MSVCR110.dll的解决方法 大致介绍一下这个错误是如何发生的.这个错误的本意是提示内存错误,但是通常情况下并不是内存存在问题,而是由于软件的问题产生了这个错误,多数情况下是DirectX 9.0的问题. 二 解决办法 1 方法:使用DirectX一键修复 请下载附件 方法2:只安装缺失的文件 可能缺少如下的几个文件:d3dx9_39.dll.d3dx9_40.dll.d3dx9_41.dl

WampServer2.0的Apache的service无法启动的解决方法

今天在家装WampServer2.0做php开发,但是装好了Apache的service却无法启动,点击了重新启动还是不 行,于是网上求助,原来才知道我开了网页迅雷的缘故,因为像qq音乐和网页迅雷这样的文件都会默认占 用80端口的,我试着把迅雷关掉重启服务,localhost启动成功,mysql测试页面也无异常,很高兴,其实还 有其他的原因,比如你是从事.net开发的肯定有IIS,那么IIS是默认强制占用80端口的,所以你需要先停止 IIS服务,具体方法是控制面板->管理工具->服务,找到II

realarm Android系统编译后内核无法启动的解决方法

由于之前版本使用的内核并非uImage格式,而在编译时使用的是非uImage格式编译,所以照成无法启动. 解决方法是,在编译内核时使用make uImage方式编译. 修改根目录下的build_realv210.sh文件,如下图所示 另外注意上图中CPU_JOB_NUM这个参数,要根据自己的电脑配置来选择,该参数在该文件的起始处设置,可以设置成电脑CPU核心数的2倍,例如:如果核心数为2,那么设置成4即可. 完整脚本下载地址:http://download.csdn.net/detail/u01

Debian9安装Atom后无法启动之解决方法

Debian9安装Atom后无法启动之解决方法 在Atom官网下载了适合于Debian的安装包后,使用$ sudo dpkg -i Atom.deb进行了安装,然后点击Atom的图标启动时,发现加载半天后退出了. 打开命令行,执行命令$ atom,发现有错误提示,提示内容如下: 大专栏  Debian9安装Atom后无法启动之解决方法ht">$ atom $ atom 依赖于libgconf-2.4.so,单并没有发现该so文件,加载失败 原来是缺少依赖了,不多说,直接修复缺少的依赖: $

关于Go语言daemon启动的方法.

昨天搞了个文件共享的小程序,遇见了意见蛋疼的事,就是启动之后终端不能关闭,不然程序也会随着关闭. 我的解决方法: nohup ./httpserver & nohup这个命令可以把程序放后台运行,顺便通过1>和2>把标准输出和标准错误重定向到文件,这样程序崩溃时才会有记录可查,这两者和程序的日志最好是分开,混在一起没办法判断轻重缓急: nohup ./server 1> server.out 2> server.err 进程启动时候记录下自己的pid: if pid :=

CentOS7.4 系统下 Tomcat 启动慢解决方法

CentOS7.4 系统下 Tomcat 启动慢解决的方法 首先查看日志信息,查看因为什么而启动慢 在CentOS7启动Tomcat时,启动过程很慢,需要几分钟,经过查看日志,发现耗时在这里:是session引起的随机数问题导致的.Tocmat的Session ID是通过SHA1算法计算得到的,计算Session ID的时候必须有一个密钥.为了提高安全性Tomcat在启动的时候会通过随机生成一个密钥. 22-Apr-2017 19:33:07.623 INFO [localhost-startS

Android Studio ADB启动失败解决方法

在用Android studio启动自己的Android代码的时候,出现adb not responding. 解决方法: 1.输入netstat -aon|findstr "5037",可以看到进程号为5196的进程(这个进程号因机器和时间而异)在占用5037端口(adb需要使用此端口). 2.打开任务管理器,选择“进程”选项卡,点击选项栏“查看-选择列...”,勾选“PID(进程标识符)”,点确定.会看到每个进程都会显示它们的PID了.找到进程号为5196的进程,结束这个进程,ki

Windows 10+Ubuntu 16.0在MBR分区上安装双系统之后没有Windows 10的启动菜单解决方法

背景: 硬盘分区方式:MBR 硬盘容量256,Windows 100,Ubuntu 156,其中主分区安装的时Windows,Ubuntu安装在逻辑分区上,文件系统为Ext4,整个Ubuntu就挂载在根目录/下,没有交换分区. 安装好之后没有Windows的启动菜单.默认进去Ubuntu. 解决方法: 1.打开grub.cfg文件 sudo gedit /boot/grub/grub.cfg 在打开的文件的最后面写入下面的内容 menuentry 'Windows 10' { set root=