用GO开发的桌面端软件如何控制只运行一个实例

我们在运行GO开发的桌面端软件时,经常遇到一个需求,就是只允许一个实例运行,比如一个游戏,我们只允许一个实例运行,防止多个实例同时运行,导致游戏卡顿。

然而如果我们不做任何处理,只要我们双击一次可执行文件,它就会运行一个新的实例,这样往往不是我们想要的结果。

怎么处理这个问题呢?

我的想法是:在程序运行时,获取到当前的进程的 PID,并将 PID 写入到用户目录下的一个临时文件里,在程序退出的时候,删除这个临时文件。

处理的逻辑一:保持旧程序运行:先读取当前运行的进程的 PID,然后读取用户目录下的临时文件,如果临时文件存在,则解析临时文件的 PID,并检查那个PID是否在运行,如果在运行,则说明程序已经启动。这时候就退出当前的程序。如果临时文件不存在,或者临时文件的 PID 不在运行,则说明程序没有启动,则将当前的PID 写入到临时文件,然后运行程序。

处理的逻辑二:运行新程序:先读取当前运行的进程的 PID,然后读取用户目录下的临时文件,如果临时文件存在,则解析临时文件的 PID,并检查那个PID是否在运行,如果在运行,则说明程序已经启动,则使用命令退出旧程序。并将当前的PID 写入到临时文件,然后运行程序。如果临时文件不存在,或者临时文件的 PID 不在运行,则说明程序没有启动,则将当前的PID 写入到临时文件,然后运行程序。

你可以根据实际情况选择使用哪种处理方式。下面我们就以逻辑二的方式来处理。

在上一篇文章中,我们用 Golang + Sciter 开发了一个简单的桌面软件,我们这回再在 main.go 中增加一段代码,来控制只运行一个实例。

var tempFile string

func checkPid() {
	pid := os.Getpid()
	tempFile = strings.TrimRight(strings.ReplaceAll(os.TempDir(), "\\", "/"), "/") + "/.googleindexing.lock"
	tmpBuf, err := os.ReadFile(tempFile)
	if err == nil {
		// 文件已存在
		tmpPid, _ := strconv.Atoi(string(tmpBuf))
		pro, err := os.FindProcess(tmpPid)
		if err == nil {
			if tmpPid > 1 {
				// 启动新的,结束旧的
				_ = pro.Kill()
			}
		}
	}
	_ = os.WriteFile(tempFile, []byte(strconv.Itoa(pid)), os.ModePerm)
}

// 在初始化的时候就判断 PID,并做相应的判断
func init() {
	checkPid()
}
  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值