Go 语言入门很简单:Go 计时器

简介: 一般来说,很多时候我们面临这样一种情况,即我们需要运行时间记录器,它不断向我们显示当前时间或在给定的时间间隔内保持执行一定的代码和平,在这种情况下,我们应该使用 Ticker,使用这个我们需要使用 go 语言的 time 包,我们有一个名为 NewTicker() 的方法,它允许我们停止和启动时间代码,我们需要通过传递 chan 和 bool 作为将使用的参数来创建一个代码通道检查它是否打开,如果通道打开意味着计时器将继续。

引言

一般来说,很多时候我们面临这样一种情况,即我们需要运行时间记录器,它不断向我们显示当前时间或在给定的时间间隔内保持执行一定的代码和平,在这种情况下,我们应该使用 Ticker,使用这个我们需要使用 go 语言的 time 包,我们有一个名为 NewTicker()  的方法,它允许我们停止和启动时间代码,我们需要通过传递 chanbool 作为将使用的参数来创建一个代码通道检查它是否打开,如果通道打开意味着计时器将继续。

Go 的计时器

Go 提供了非常简单的语法来实现一个计时器,定时器的结构体定义:

type Ticker struct {
  C <-chan Time  // 抛出来的channel,给上层系统使用,实现计时
  r runtimeTimer  // 给系统管理使用的定时器,系统通过该字段确定定时器是否到时,如果到时,调用对应的函数向C中推送当前时间。
} 


Ticker 对外仅暴露一个 channel,指定的时间到来时就往该 channel 中写入系统时间,也即一个事件。


Ticker 的使用方式也很简单,代码如下:

import time
TimeTicker := time.NewTicker(1 * time.Second)
TimeTicker.Stop()


  • time : 如果想要使用 time 中的计时器,那么需要将其导入代码中
  • NewTicker() 函数:这个很好理解,新建一个计时器,然后该计时器以时间表达式作为参数,比如一秒 1*time.Second ,而不是直接传入 1 ,也可以传入其他时间方式,比如 2*time.Millisecond
  • 最后,我们可以使用时间表达式调用由 NewTicker 函数创建的变量上的任何活动。例如,在上面的语法中,我们使用了TimeTicker.Stop() 来停止时间计时器。我们可以在特定条件下使用它,比如定义倒数 10 秒的计时器,就可以检查时间是否满足 10 秒,一旦 10 秒发生,我们可以调用TimeTicker.stop()


我们可以每隔一秒输出一个结果,比如实现一个倒数 10 个数的功能:

package main
import (
  "fmt"
  "time"
)
func main() {
  TimeTicker := time.NewTicker(1 * time.Second)
  i := 10
  for {
    <-TimeTicker.C
    fmt.Println("i = ", i)
    i--
    if i == 0 {
      TimeTicker.Stop()
      break
    }
  }
}


然后执行该程序:

$ go run main.go
i =  10
i =  9
i =  8
i =  7
i =  6
i =  5
i =  4
i =  3
i =  2
i =  1

Ticker 计时器是如何工作的?

NewTicker创建的计时器与NewTimer创建的计时器持有的时间channel一样都是带一个缓存的channel,每次触发后执行的函数也是sendTime,这样即保证了无论有误接收方Ticker触发时间事件时都不会阻塞:

func NewTicker(d Duration) *Ticker {
    if d <= 0 {
        panic(errors.New("non-positive interval for NewTicker"))
    }
    // Give the channel a 1-element time buffer.
    // If the client falls behind while reading, we drop ticks
    // on the floor until the client catches up.
    c := make(chan Time, 1)
    t := &Ticker{
        C: c,
        r: runtimeTimer{
            when:   when(d),
            period: int64(d),
            f:      sendTime,
            arg:    c,
        },
    }
    startTimer(&t.r)
    return t
}


NewTicker()只是构造了一个 Ticker,然后把 Ticker.r 通过 startTimer()交给系统协程维护。其中 period 为事件触发的周期。


其中sendTime()方法便是定时器触发时的动作:

func sendTime(c interface{}, seq uintptr) {
    select {
    case c.(chan Time) <- Now():
    default:
    }
}


sendTime  接收一个管道作为参数,其主要任务是向管道中写入当前时间。


停止 Ticker:


停止 Ticker,只是简单的把 Ticker 从系统协程中移除。函数主要实现如下:

func (t *Ticker) Stop() {
    stopTimer(&t.r)
}


stopTicker() 即通知系统协程把该 Ticker 移除,即不再监控。系统协程只是移除 Ticker 并不会关闭管道,以避免用户协程读取错误。

Ticker 使用方式

例子一:

package main
import (
  "fmt"
  "time"
)
func main() {
  TimeTicker := time.NewTicker(3 * time.Second)
  tickerChannel := make(chan bool)
  go func() {
    for {
      select {
      case timeticker := <-TimeTicker.C:
        fmt.Println("The time for current is : ", timeticker)
      case <-tickerChannel:
        return
      }
    }
  }()
  time.Sleep(6 * time.Second)
  TimeTicker.Stop()
  tickerChannel <- true
  fmt.Println("Time for running ticker is completed")
}


运行该代码:

$ go run main.go
The time for current is :  2022-04-29 22:37:13.93862 +0800 CST m=+3.000267823
The time for current is :  2022-04-29 22:37:16.939081 +0800 CST m=+6.000707515
Time for running ticker is completed


例子二:

package main
import (
  "fmt"
  "time"
)
func main() {
  tm := time.Millisecond
  tickerTicker := time.NewTicker(400 * tm)
  tickerChaneel := make(chan bool)
  go func() {
    for {
      select {
      case <-tickerChaneel:
        return
      case tmtr := <-tickerTicker.C:
        fmt.Println("Ticker time at current is", tmtr)
      }
    }
  }()
  time.Sleep(1400 * time.Millisecond)
  tickerTicker.Stop()
  tickerChaneel <- true
  fmt.Println("Ticker has stopped now")
}

运行该代码:

$ go run main.go         
Ticker time at current is 2022-04-29 22:39:51.13057 +0800 CST m=+0.400159916
Ticker time at current is 2022-04-29 22:39:51.531516 +0800 CST m=+0.801102997
Ticker time at current is 2022-04-29 22:39:51.931238 +0800 CST m=+1.200822301
Ticker has stopped now

总结

本文简单了解了 go 计时器 Ticker 的基本概念,介绍了它的工作原理,并且我们专注于 go 语言中 ticker  的语法和使用。最后展现了 go 语言中的一些重要示例。

相关文章
|
6天前
|
存储 JSON 监控
Viper,一个Go语言配置管理神器!
Viper 是一个功能强大的 Go 语言配置管理库,支持从多种来源读取配置,包括文件、环境变量、远程配置中心等。本文详细介绍了 Viper 的核心特性和使用方法,包括从本地 YAML 文件和 Consul 远程配置中心读取配置的示例。Viper 的多来源配置、动态配置和轻松集成特性使其成为管理复杂应用配置的理想选择。
23 2
|
4天前
|
Go 索引
go语言中的循环语句
【11月更文挑战第4天】
13 2
|
4天前
|
Go C++
go语言中的条件语句
【11月更文挑战第4天】
16 2
|
8天前
|
程序员 Go
go语言中的控制结构
【11月更文挑战第3天】
85 58
|
7天前
|
监控 Go API
Go语言在微服务架构中的应用实践
在微服务架构的浪潮中,Go语言以其简洁、高效和并发处理能力脱颖而出,成为构建微服务的理想选择。本文将探讨Go语言在微服务架构中的应用实践,包括Go语言的特性如何适应微服务架构的需求,以及在实际开发中如何利用Go语言的特性来提高服务的性能和可维护性。我们将通过一个具体的案例分析,展示Go语言在微服务开发中的优势,并讨论在实际应用中可能遇到的挑战和解决方案。
|
4天前
|
Go
go语言中的 跳转语句
【11月更文挑战第4天】
13 4
|
4天前
|
JSON 安全 Go
Go语言中使用JWT鉴权、Token刷新完整示例,拿去直接用!
本文介绍了如何在 Go 语言中使用 Gin 框架实现 JWT 用户认证和安全保护。JWT(JSON Web Token)是一种轻量、高效的认证与授权解决方案,特别适合微服务架构。文章详细讲解了 JWT 的基本概念、结构以及如何在 Gin 中生成、解析和刷新 JWT。通过示例代码,展示了如何在实际项目中应用 JWT,确保用户身份验证和数据安全。完整代码可在 GitHub 仓库中查看。
18 1
|
8天前
|
Go 数据处理 API
Go语言在微服务架构中的应用与优势
本文摘要采用问答形式,以期提供更直接的信息获取方式。 Q1: 为什么选择Go语言进行微服务开发? A1: Go语言的并发模型、简洁的语法和高效的编译速度使其成为微服务架构的理想选择。 Q2: Go语言在微服务架构中有哪些优势? A2: 主要优势包括高性能、高并发处理能力、简洁的代码和强大的标准库。 Q3: 文章将如何展示Go语言在微服务中的应用? A3: 通过对比其他语言和展示Go语言在实际项目中的应用案例,来说明其在微服务架构中的优势。
|
8天前
|
Go 数据处理 调度
探索Go语言的并发模型:Goroutines与Channels的协同工作
在现代编程语言中,Go语言以其独特的并发模型脱颖而出。本文将深入探讨Go语言中的Goroutines和Channels,这两种机制如何协同工作以实现高效的并发处理。我们将通过实际代码示例,展示如何在Go程序中创建和管理Goroutines,以及如何使用Channels进行Goroutines之间的通信。此外,本文还将讨论在使用这些并发工具时可能遇到的常见问题及其解决方案,旨在为Go语言开发者提供一个全面的并发编程指南。
|
6天前
|
Go 调度 开发者
探索Go语言中的并发模式:goroutine与channel
在本文中,我们将深入探讨Go语言中的核心并发特性——goroutine和channel。不同于传统的并发模型,Go语言的并发机制以其简洁性和高效性著称。本文将通过实际代码示例,展示如何利用goroutine实现轻量级的并发执行,以及如何通过channel安全地在goroutine之间传递数据。摘要部分将概述这些概念,并提示读者本文将提供哪些具体的技术洞见。