浅析Go使用定时器时如何避免潜在的内存泄漏陷阱

 更新时间:2024年01月28日 08:12:55   作者:波罗学  
这篇文章来和大家一起探讨一下Go 中如何高效使用 timer,特别是与select 一起使用时,如何防止潜在的内存泄漏问题,感兴趣的可以了解下

引出问题

先看一个例子,我们在 Go 中的 select 使用定时器,实现为消息监听加上超时能力。

核心代码,如下所示:

func main() {
  ch := make(chan int)
  // 启动一个goroutine
  go func() {
    for {
      select {
      case num := <-ch:
        fmt.Println("获取到的数字是", num)
      case <-time.After(2 * time.Second):
        fmt.Println("时间到了!!!")
      }
    }
  }()
  for i := 0; i < 5; i++ {
    ch <- i
    time.Sleep(1 * time.Second)
  }
}

在这个例子中,select 语句用于监听 channel 消息和超时。然而,我要关注的重点是 timer 的行为。它是不是能达到我们预期的目标呢?为消息监听加上超时效果呢?

检查定时器行为

如果运行这段代码,将会发现,如果 timer 设置为 2 秒,主循环设置 1 秒的延迟时间,timer 不会触发。

如下是程序的运行输出:

获取到的数字是 0
获取到的数字是 1
获取到的数字是 2
获取到的数字是 3
获取到的数字是 4

这是因为每次循环,time.After 创建都会返回一个新的定时器,产生的后果就是,每次多会重置 select 调用的时间。

相反,如果将定时器的超时设置为 1 秒,将主循环的time.Sleep设置为 2 秒,就能触发定时器,输出 "时间到了!!!"。这证明了这个定时器是有效运行的。

潜在的内存泄漏

Go标准库文档提到,每次调用time.After都会创建一个新的定时器。然而,我们需要认真考虑一个重要问题。

来自官方文档引用:

The underlying Timer is not recovered by the garbage collector until the timer fires.

如果这些 timer 没有达到设定时间,就不会被 GC。这会导致内存泄漏。毫无疑问,如果在常驻程序中频繁使用 timer 的,内存泄漏将会日积月累。

最佳实践

要高效地管理资源并避免 timer 的内存泄漏,建议使用 time.NewTimer 和 timer.Reset 组合。这种方法允许重复使用一个定时器,减少资源消耗和潜在的内存泄漏风险。

例如,如下是使用 time.NewTimer 改进的代码示例:

// 为定时器定义持续时间。
idleDuration := 5 * time.Minute
// 使用指定的持续时间创建新的定时器。
idleDelay := time.NewTimer(idleDuration)
// 确保定时器适当地停止以避免资源泄漏。
defer idleDelay.Stop()
// 进入循环以处理传入的消息或基于时间的事件。
for {
  // 在每次循环迭代开始时重置定时器到指定的持续时间。
  idleDelay.Reset(idleDuration)
  
  // 使用select等待多个通道操作。
  select {
  // 处理传入消息的情况。
  case s, ok := <-in:
    // 检查通道是否关闭。如果是,退出循环。
    if !ok {
      return
    }
    // 处理接收到的消息`s`。
    // 在这里添加相关代码来处理消息。
  // 处理定时器超时的情况。
  case <-idleDelay.C:
    // 增加空闲计数器或处理超时事件。
    // 这通常是您会在这里添加代码来处理超时情况的地方。
    idleCounter.Inc()
  // 处理取消或上下文过期的情况。
  case <-ctx.Done():
    // 如果上下文已完成,则退出循环。
    return
  }
}

流程如下所示:

这里例子中演示了 Go 语言中如何正确使用和管理 timer。通过遵循 Go 标准库的建议将能产出更高效和可靠的程序。

结论

本文通过一个代码案例演示了 GO 中 timer.After 可能产生的潜在内存泄漏问题。通过使用官方推荐的方案,利用重置定时器时间实现 Timer 的重复利用,避免了潜在的内存泄漏问题。

到此这篇关于浅析Go使用定时器时如何避免潜在的内存泄漏陷阱的文章就介绍到这了,更多相关Go内存泄漏内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 浅析Go语言编程当中映射和方法的基本使用

    浅析Go语言编程当中映射和方法的基本使用

    这篇文章主要介绍了浅析Go语言编程当中映射和方法的基本使用,是golang入门学习中的基础知识,需要的朋友可以参考下
    2015-10-10
  • 浅析Go语言中包的介绍与初始化

    浅析Go语言中包的介绍与初始化

    这篇文章主要为大家详细介绍了Go语言中包的介绍与初始化,从而搞清Go程序的执行次序,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起了解下
    2023-10-10
  • 利用systemd部署golang项目的实现方法

    利用systemd部署golang项目的实现方法

    这篇文章主要介绍了利用systemd部署golang项目的实现方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-11-11
  • GO语言常用的文件读取方式

    GO语言常用的文件读取方式

    这篇文章主要介绍了GO语言常用的文件读取方式,涉及一次性读取、分块读取与逐行读取等方法,是非常实用的技巧,需要的朋友可以参考下
    2014-12-12
  • Golang性能优化的技巧分享

    Golang性能优化的技巧分享

    性能优化的前提是满足正确可靠、简洁清晰等质量因素,针对 Go语言特性,本文为大家整理了一些Go语言相关的性能优化建议,感兴趣的可以了解一下
    2023-07-07
  • Go语言微服务中实现链路追踪

    Go语言微服务中实现链路追踪

    在微服务架构中,链路追踪技术可以帮助我们跟踪请求在各个服务之间的传播路径,本文就来介绍一下Go语言微服务中实现链路追踪,感兴趣的可以了解一下
    2024-12-12
  • Go代码检查工具golangci-lint安装使用方法

    Go代码检查工具golangci-lint安装使用方法

    这篇文章主要给大家介绍了关于Go代码检查工具golangci-lint安装使用的相关资料,golangci-lint用于许多开源项目中,比如kubernetes、Prometheus、TiDB等都使用golangci-lint用于代码检查,需要的朋友可以参考下
    2024-01-01
  • Go语言异常处理案例解析

    Go语言异常处理案例解析

    这篇文章主要介绍了Go语言异常处理案例解析,本篇文章通过简要的案例,讲解了该项技术的了解与使用,以下就是详细内容,需要的朋友可以参考下
    2021-07-07
  • gorm RowsAffected()返回0的问题及解决

    gorm RowsAffected()返回0的问题及解决

    在gorm中,`RowsAffected()`方法用于获取更新操作的受影响行数,如果在执行更新操作后立即调用`RowsAffected()`,可能会得到0,因为该方法在内部已经执行了数据库操作并更新了数据,正确的使用方法是在执行更新操作后,通过返回的`db`对象来获取受影响的行数
    2025-12-12
  • Golang 错误捕获Panic与Recover的使用

    Golang 错误捕获Panic与Recover的使用

    对于Go语言的错误是通过返回值的方式,本文主要介绍了Golang 错误捕获Panic与Recover的使用,文中根据实例编码详细介绍的十分详尽,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-03-03

最新评论