go 语言中的定时器

       在日常的开发过程中,我们免不了与定时器打交道。通常需要定时执行某个程序,可能是每隔一段时间执行一次,也可能是固定的每天在某一个时间点执行一次。总之,定时器的使用是非常频繁的,下面将介绍几种常用定时器。

1. 假设每隔三秒,我们打印一段信息,可以采用如下方式。我们将要执行的程序放在一个go协程中去执行。

package main

import (
  "fmt"
  "time"
)

func main() {

   go func() {
       for {
          fmt.Println("hello")
          time.Sleep(time.Second * 3)
       }
   }()
   time.Sleep(time.Second * 60)
}

2. 如果需要在某个某个固定时间打印某一信息,可采用如下方式:其中第一个协程不断的检查当前时间是否是我们所记录的目标时间,如果是,则该协程向管道nahc写入一个bool类型的值;第二个协程不断的检查管道nahc中是否存在值,如果存在,则说明到达我们所规定的目标时间,于是打印某一信息。

package main

import (
   "time"
   "sync"
   "fmt"
)

var (
   t = "22:51:00"
   nahc chan bool  
)

func init() {
   nahc = make(chan bool)
}

func main() {
   var wg sync.WaitGroup
   wg.Add(2)
   go func() {
       defer wg.Done()
       for {
           fmt.Println(t)
           now := time.Now()
           location, _ := time.LoadLocation("Local")
           aimTime, _ := time.ParseInLocation("2006-01-02 15:04:05", fmt.Sprintf("%s %s", now.Format("2006-01-02"), t), location)
           if now.After(aimTime) {
              aimTime, _ = time.ParseInLocation("2006-01-02 15:04:05", fmt.Sprintf("%s %s", now.AddDate(0, 0, 1).Format("2006-01-02"), t), location)
           }
           duration := aimTime.Sub(now)
           select {
              case <-time.After(duration):
              nahc <- true
           }
       }
   }()

   go func() {
      defer wg.Done()
      for range nahc {
      	fmt.Println("hello")
      }
   }()
   wg.Wait()
}

3. 下面这个定时器采用了time包的NewTicker函数。

report_ticker := time.NewTicker(time.Second * time.Duration(3))
go func() {
   for range report_ticker.C {
       go fmt.Println("hello")
   }
}()

 

### Go语言定时器的使用方法与实现机制 #### 1. 定时器的基础概念 Go语言提供了一次性定时器 `Timer` 和周期性定时器 `Ticker`,用于处理延迟操作和周期性任务。这两种定时器均基于通道(channel)来传递时间信号[^1]。 #### 2. 一次性定时器 Timer 的使用 `Timer` 是一种只触发一次的时间管理工具。它通过创建一个带缓冲区大小为1的通道返回当前时间戳,当设定的时间到达后,可以通过读取通道获取通知。 以下是 `Timer` 的基本用法: ```go package main import ( "fmt" "time" ) func main() { timer := time.NewTimer(2 * time.Second) // 创建一个2秒的一次性定时器 select { case <-timer.C: // 等待定时器到期 fmt.Println("定时器已到期") } } ``` 上述代码展示了如何设置一个两秒钟后触发的通知,并打印消息。如果需要取消未到期的定时器,可以调用 `Stop()` 方法[^1]。 #### 3. 周期性定时器 Ticker 的使用 `Ticker` 提供了按固定间隔发送事件的功能。它的核心是一个无限循环,每隔指定时间段向通道写入当前时间。需要注意的是,`Ticker` 需要手动停止以释放资源,否则可能导致内存泄漏。 下面是一段关于 `Ticker` 的示例代码: ```go package main import ( "fmt" "time" ) func main() { ticker := time.NewTicker(500 * time.Millisecond) // 每隔500毫秒触发一次 defer ticker.Stop() for i := 0; i < 5; i++ { // 循环五次接收并打印时间 select { case t := <-ticker.C: fmt.Printf("接收到时间:%v\n", t) } } fmt.Println("完成") } ``` 这段程序每半秒输出一次当前时间,总共执行五次之后退出[^4]。 #### 4. 定时器内部实现原理 无论是 `Timer` 还是 `Ticker`,它们底层都依赖于相同的 `runtime.Timer` 数据结构[^3]。具体来说: - **数据结构** `runtime.Timer` 结构体定义了每个定时器的核心属性,包括触发时间和周期长度等字段。这些信息会被存储到全局最小堆中以便高效调度。 - **工作流程** - 所有活动中的定时器都会加入到由运行时维护的一个优先级队列里; - 单独启动的 goroutine 不断轮询最早应该触发的那个定时器是否已经到了预定时刻; - 到达目标时间节点后立即唤醒对应的任务逻辑[^3]。 这种设计使得即使存在大量不同超时时长的需求也能保持较低开销。 --- ###
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值