go 中,进程退出如何保障异步协程的完整退出。

2022-10-23 14:44:17 +08:00
 ml444

写了一个 go 的异步 log 组件。为了保障异步的 channel 被彻底消费完,我增加了Exit()函数,但我想让调用无感知,不用刻意去调用这个函数。有没有什么办法或方式能够实现?

package main

import (
	"os"
	
	"github.com/ml444/glog"
)

func main() {
	// doing something
	log.Info("hello world")
	// doing something

	_ = log.Exit()
2126 次点击
所在节点    Go 编程语言
13 条回复
0o0O0o0O0o
2022-10-23 14:53:42 +08:00
不能,一般用 defer 吧

defer log.Exit()
VANHOR
2022-10-23 14:58:43 +08:00
确实得有一个地方来判断 channel 是否被消耗完了,所以还是得写上的。
iyaozhen
2022-10-23 15:09:47 +08:00
理论上不能,所以很多时候需要在 main.go 加个 defer log.flush
securityCoding
2022-10-23 17:25:36 +08:00
可以在 new 方法内部,启动一个协程监听系统退出信号调用 exit
Morriaty
2022-10-23 18:00:30 +08:00
官方做法是 sync.WaitGroup ,主要三个函数

group.Add(1) 每进入一个协程加一次计数
defer group.Done() 协程结束标志,计数减一
group.Wait() 等待计数清零,约等于你这里的 log.Exit()
fuis
2022-10-23 18:56:13 +08:00
jackliang007
2022-10-24 00:01:35 +08:00
日志全部发送到消息队列,然后用开个协程来处理。😄
bugfan
2022-10-24 01:15:00 +08:00
4 楼正解
ml444
2022-11-04 17:51:51 +08:00
@securityCoding 没用哦,我试过 panic 时,是收不到信号的
securityCoding
2022-11-04 18:16:07 +08:00
业务协程 panic 要他自己兜底 ,跟你的日志组件有啥关系?
ml444
2022-11-04 18:19:15 +08:00
@securityCoding panic 会导致异步日志丢失,所以我问的是有什么方式能够保障 panic 这种异常故障的时候,让日志能够写完再退出
securityCoding
2022-11-04 18:51:39 +08:00
@ml444 没理解你的意思,一般框架层都会做 recover 兜底,同时日志组件也需要定时 flush
securityCoding
2022-11-04 18:53:45 +08:00
@ml444 刷盘思路可以看下 zap 或者 gozero log

这是一个专为移动设备优化的页面(即为了让你能够在 Google 搜索结果里秒开这个页面),如果你希望参与 V2EX 社区的讨论,你可以继续到 V2EX 上打开本讨论主题的完整版本。

https://www.v2ex.com/t/889105

V2EX 是创意工作者们的社区,是一个分享自己正在做的有趣事物、交流想法,可以遇见新朋友甚至新机会的地方。

V2EX is a community of developers, designers and creative people.

© 2021 V2EX