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 语言中的一些重要示例。

相关文章
|
1天前
|
Java 编译器 Go
探索Go语言的性能优化技巧
在本文中,我们将深入探讨Go语言的底层机制,以及如何通过代码层面的优化来提升程序性能。我们将讨论内存管理、并发控制以及编译器优化等关键领域,为你提供一系列实用的技巧和最佳实践。
|
1天前
|
Cloud Native Go API
Go语言在微服务架构中的创新应用与实践
本文深入探讨了Go语言在构建高效、可扩展的微服务架构中的应用。Go语言以其轻量级协程(goroutine)和强大的并发处理能力,成为微服务开发的首选语言之一。通过实际案例分析,本文展示了如何利用Go语言的特性优化微服务的设计与实现,提高系统的响应速度和稳定性。文章还讨论了Go语言在微服务生态中的角色,以及面临的挑战和未来发展趋势。
|
1天前
|
安全 Go 调度
探索Go语言的并发模式:协程与通道的协同作用
Go语言以其并发能力闻名于世,而协程(goroutine)和通道(channel)是实现并发的两大利器。本文将深入了解Go语言中协程的轻量级特性,探讨如何利用通道进行协程间的安全通信,并通过实际案例演示如何将这两者结合起来,构建高效且可靠的并发系统。
|
1天前
|
安全 Go 开发者
破译Go语言中的并发模式:从入门到精通
在这篇技术性文章中,我们将跳过常规的摘要模式,直接带你进入Go语言的并发世界。你将不会看到枯燥的介绍,而是一段代码的旅程,从Go的并发基础构建块(goroutine和channel)开始,到高级模式的实践应用,我们共同探索如何高效地使用Go来处理并发任务。准备好,让Go带你飞。
|
2天前
|
运维 Go 开发者
Go语言在微服务架构中的应用与优势
本文深入探讨了Go语言在构建微服务架构中的独特优势和实际应用。通过分析Go语言的核心特性,如简洁的语法、高效的并发处理能力以及强大的标准库支持,我们揭示了为何Go成为开发高性能微服务的首选语言。文章还详细介绍了Go语言在微服务架构中的几个关键应用场景,包括服务间通信、容器化部署和自动化运维等,旨在为读者提供实用的技术指导和启发。
|
2天前
|
安全 Go 调度
探索Go语言的并发之美:goroutine与channel
在这个快节奏的技术时代,Go语言以其简洁的语法和强大的并发能力脱颖而出。本文将带你深入Go语言的并发机制,探索goroutine的轻量级特性和channel的同步通信能力,让你在高并发场景下也能游刃有余。
|
3天前
|
Go 开发者
Go语言中的并发编程:从基础到实践
在当今的软件开发中,并发编程已经成为了一项不可或缺的技能。Go语言以其简洁的语法和强大的并发支持,成为了开发者们的首选。本文将带你深入了解Go语言中的并发编程,从基础概念到实际应用,帮助你掌握这一重要的编程技能。
|
3天前
|
安全 Go 调度
探索Go语言的并发模型:Goroutine与Channel的魔力
本文深入探讨了Go语言的并发模型,不仅解释了Goroutine的概念和特性,还详细讲解了Channel的用法和它们在并发编程中的重要性。通过实际代码示例,揭示了Go语言如何通过轻量级线程和通信机制来实现高效的并发处理。
|
3天前
|
存储 安全 Go
Go语言切片:从入门到精通的深度探索###
本文深入浅出地剖析了Go语言中切片(Slice)这一核心概念,从其定义、内部结构、基本操作到高级特性与最佳实践,为读者提供了一个全面而深入的理解。通过对比数组,揭示切片的灵活性与高效性,并探讨其在并发编程中的应用优势。本文旨在帮助开发者更好地掌握切片,提升Go语言编程技能。 ###
【Go语言入门100题】026 I Love GPLT (5 分) Go语言 | Golang
L1-026 I Love GPLT (5 分) Go语言|Golang 这道超级简单的题目没有任何输入。 你只需要把这句很重要的话 —— “I Love GPLT”——竖着输出就可以了。 所谓“竖着输出”,是指每个字符占一行(包括空格),即每行只能有1个字符和回车。
596 0