文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

如何让Go程序以后台进程或daemon方式运行

2024-11-30 01:51

关注

我在网上找到了一个开源库,github.com/sevlyar/go-daemon,它很方便地实现了在后台启动一个新的进程,但如果后台进程再次尝试作为另一个后台进程启动,会出现错误。

后来我阅读了源代码才发现:为了区分当前进程是父进程还是子进程,作者巧妙地设计了一个环境变量标识。正是因为这种识别策略,该库只能启动一次自身作为后台进程,无法连续启动自身为后台进程。

不过,这种使用环境变量来区分进程身份的思路给我启发很大。基于这个想法,我通过延伸和优化,最终实现了在保持参数不变的情况下连续启动自身为后台进程。我对作者表示敬意。

此外,我还找到了一些其他的库,它们的思路有所不同,主要通过添加特殊参数来标记进程身份。但是,这些方法并没有完美地解决让进程启动自身的问题,令我有些遗憾。

最终,我决定自己实现一个库来解决我的项目需求,并希望它是一个通用的库,可以快速方便地将用Go语言编写的服务程序转为后台运行或守护进程模式运行。本文总结了我在这次探索中的经验和收获。

首先,让我们区分一下两个概念:后台运行和守护进程。平常交流时,我们可能不太区分或区分不够清晰。在本文中,我想明确如下定义:

后台运行:指进程在操作系统中以非显示方式运行,没有与任何命令行终端或程序界面相关联。这种方式下运行的进程称为后台进程,比如没有与任何终端相关联的命令行程序进程。

守护进程:也称为守护进程,它首先以后台运行方式启动,然后还有额外的职责。在本文中,我的定义是守护进程可以监视Go服务程序进程的状态,如果异常退出,可以自动重新启动。这样守护进程可以确保服务程序一直在后台运行,即使它在某些情况下崩溃或意外终止。

接下来,我将介绍如何使用Go代码来实现在后台运行的程序,并将其转化为一个守护进程。

后台运行程序

要将Go程序在后台运行,可以使用一些操作系统级别的方法。以下是一种简单的方法:

package main

import (
    "fmt"
    "os"
    "os/exec"
    "syscall"
)

func main() {
    if os.Getppid() != 1 {
        cmd := exec.Command(os.Args[0])
        cmd.Start()
        fmt.Println("Background process ID:", cmd.Process.Pid)
        os.Exit(0)
    }

    // 在这里写入具体的业务逻辑代码
    fmt.Println("Running in background...")
    select {}
}

在上面的代码中,我们首先使用os.Getppid()函数获取当前进程的父进程ID。如果父进程不是1,说明当前进程不是守护进程,而是从终端启动的。在这种情况下,我们创建一个新的命令,使用相同的参数再次启动程序,并在后台运行。我们打印出新进程的PID,并退出初始进程。

如果进程的父进程是1,那么说明当前进程已经是守护进程了,我们可以在此处写入具体的业务逻辑代码。

使用这种方法,我们可以确保程序在后台运行,而且还可以检查是否已经启动了一个后台进程。

守护进程

将程序转化为守护进程需要额外的步骤,我们需要创建一个监听子进程状态的循环,并在子进程异常退出时重新启动它。以下是一个简单的守护进程实现:

package main

import (
    "fmt"
    "os"
    "os/exec"
    "syscall"
)

func main() {
    if os.Getppid() != 1 {
        cmd := exec.Command(os.Args[0])
        cmd.Start()
        fmt.Println("Background process ID:", cmd.Process.Pid)
        os.Exit(0)
    }

    // 在这里写入具体的业务逻辑代码
    fmt.Println("Running in background...")

    for {
        cmd := exec.Command(os.Args[0])
        cmd.Start()
        exitCh := make(chan error)
        go func() {
            exitCh <- cmd.Wait()
        }()

        err := <-exitCh
        if err != nil {
            fmt.Println("Process exited with error:", err)
        } else {
            fmt.Println("Process exited successfully")
        }

        select {
        case <-exitCh:
        default:
        }
    }
}

在上面的代码中,我们添加了一个循环,用于监听子进程的状态。在每次子进程退出之后,我们使用相同的参数再次启动守护进程,并重新开始监听。这样就可以确保服务程序在异常退出时能够自动重新启动。

这只是一个简单的守护进程实现,你可以根据自己的需求进行扩展和优化。

来源:Go语言圈内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯