如何更酷地实现 Go 程序热开关功能
在介绍新方案之前,我们回顾一下,在 Go 程序中,pprof 是如何做的。
接口调用
对程序进行性能采样,可能会影响到它的服务能力。因此,线上采样一般是在指定的小块时间范围内进行,需要有效的开关控制。
为了做到这点,我们一般采用在代码中引入 net/http/pprof 包(其 init 函数绑定了路由功能函数)的方式。外部访问指定端口的 HTTP 服务时开启采样功能,采样时间结束后,即关闭采集。
实现代码如下所示
package main import ( "net/http" _ "net/http/pprof" ) func main() { go func() { _ = http.ListenAndServe(":8080", nil) }() ... }
这种做法,当然没什么问题。我们可以学习它,将其他的开关功能也做成 HTTP 服务。但,还有其他更酷的方式吗?
信号通知
在信号处理与 Go 程序的优雅退出一文中,我们谈论过信号机制,它用以向应用程序发送某种事件通知。
我们可以将基于接口触发的方式改为信号通知。
首先,构造采样功能函数(对应于 net/http/pprof 包下 init 函数中绑定的路由功能函数)。
func RegisterSignalForProfiling(sig os.Signal) { ch := make(chan os.Signal) started := false signal.Notify(ch, sig) go func() { var memoryProfile, cpuProfile, traceProfile *os.File for range ch { if started { pprof.StopCPUProfile() trace.Stop() pprof.WriteHeapProfile(memoryProfile) memoryProfile.Close() cpuProfile.Close() traceProfile.Close() started = false } else { cpuProfile, _ = os.Create("cpu.pprof") memoryProfile, _ = os.Create("memory.pprof") traceProfile, _ = os.Create("runtime.trace") pprof.StartCPUProfile(cpuProfile) trace.Start(traceProfile) started = true } } }() }
在上述函数中,我们定义了接收信号通道<span style="font-size: 15px;">ch</span>
,通过<span style="font-size: 15px;">signal.Notify(ch, sig)</span>
将指定的通知信号<span style="font-size: 15px;">sig</span>
与<span style="font-size: 15px;">ch</span>
进行绑定。<span style="font-size: 15px;">for range ch</span>
将阻塞等待外部信号<span style="font-size: 15px;">sig</span>
,随着<span style="font-size: 15px;">sig</span>
信号的到来,交替进入开启或关闭采样的逻辑。
在<span style="font-size: 15px;">main</span>
函数中,就可以这样替代<span style="font-size: 15px;">http.ListenAndServe(":8080", nil)</span>
了。
package main import ( "syscall" ... ) func main() { RegisterSignalForProfiling(syscall.Signal(31)) ... }
在 linux 系统,可以通过<span style="font-size: 15px;">kill -signal_number pid</span>
命令向程序发送指定信号。
如上代码所示,我们硬编码指定的采样开关信号值是 31。因此,当程序运行起来后,我们在控制台输入<span style="font-size: 15px;">kill -31 pid</span>
命令,即可开启采样,再次输入<span style="font-size: 15px;">kill -31 pid</span>
命令,就关闭了采样。
依葫芦画瓢,我们再来一个打印 goroutine 堆栈信息的热开关函数,是不是很酷?
func RegisterSignalForPrintStack(sig os.Signal) { ch := make(chan os.Signal) signal.Notify(ch, sig) go func() { for range ch { buffer := make([]byte, 1024*1024*4) runtime.Stack(buffer, true) fmt.Println(string(buffer)) } }() }
总结
热开关是一个很简单常用的功能,无非是选择何种触发与等待方式。基于接口的调用更适合于远程控制,基于信号则便于本地控制。
以上是如何更酷地实现 Go 程序热开关功能的详细内容。更多信息请关注PHP中文网其他相关文章!

热AI工具

Undresser.AI Undress
人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover
用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool
免费脱衣服图片

Clothoff.io
AI脱衣机

AI Hentai Generator
免费生成ai无尽的。

热门文章

热工具

记事本++7.3.1
好用且免费的代码编辑器

SublimeText3汉化版
中文版,非常好用

禅工作室 13.0.1
功能强大的PHP集成开发环境

Dreamweaver CS6
视觉化网页开发工具

SublimeText3 Mac版
神级代码编辑软件(SublimeText3)

热门话题

开发中,我们经常会有热开关的需求,即特定功能在程序运行中的适当时候对它进行打开或关闭。例如性能分析中使用的 pprof 采样,就是一种典型的热开关。本文将讨论如何将这种热开关做得更酷。

获取应用程序的运行指标,可以让我们更好地了解它的实际状况。将这些指标对接到 prometheus、zabbix 等监控系统,能够对应用程序持续检测,发现异常可以及时告警并得到处理。

Go是一种受欢迎的编程语言,与其他编程语言相比,Go的编译速度较快,内存消耗较少。但是,有时候我们的Go程序会因为缺少依赖而导致编译失败。那么,为什么会发生这种情况呢?首先,我们需要了解一下Go编译的原理。Go是静态编译型语言,即在编译时就将程序翻译成机器码,然后直接运行。与动态编译型语言相比,Go的编译过程更为复杂,因为在编译之前,需要将所有要使用的包都转

最近,越来越多的人开始使用GoQUIC来构建网络应用。由于其高效的传输性能和可靠性,GoQUIC已经成为许多项目的首选。但是,在实际使用过程中,一些开发者发现他们的Go程序无法正确使用GoQUIC库。下面,我们就来分析一下可能导致Go程序无法正常使用GoQUIC库的原因。1.版本问题首先,你需要确定你的GoQUIC版本是否是最新的。GoQUIC经常更新,如果

在公司的不断发展中,一开始大多是大单体,改造慢了,一个仓库会有使用十几年的情况,仓库的规模基本是不断增大的过程。

Go是一种高效的编程语言,提供了内存管理方面的特殊机制。然而,即使在使用这种语言的时候也有可能出现一些问题,比如“outofmemory”错误。那么,为什么我的Go程序会出现这种错误呢?内存泄漏内存泄漏是一种常见的问题,它在Go语言中同样存在。当你的Go程序分配了大量的内存,并在执行某些操作后没有彻底释放这些内存时,就会出现内存泄漏问题。如果出现内存泄漏

Golang(Go)是一门语言,它非常擅长处理错误和异常情况。与其他语言不同,Go通过简单而有效的错误处理机制来处理异常情况。尽管Go的错误处理机制非常强大和灵活,但某些程序员在程序中实现错误处理时仍会遇到麻烦。这篇文章旨在帮助解决关于为什么Go程序中的异常处理无效的问题,以及如何正确地处理异常情况。在Go中无效的异常处理通常是因为程序员未正确处理错误或者误

在使用Go语言进行开发过程中,难免会遇到各种各样的错误。其中一种常见的错误就是“coredumped”,这个错误消息可能令一些开发者很困惑。本文将讲解这个错误的原因以及如何解决它。“coredumped”的含义在Linux操作系统中,“coredumped”是一种错误消息,它表示一个进程在执行过程中意外退出,并且已经生成了一个所谓的“core”文件。这
