iis服务器助手广告广告
返回顶部
首页 > 资讯 > 前端开发 > node.js >怎么理解并掌握的Go高级并发模式计时器
  • 609
分享到

怎么理解并掌握的Go高级并发模式计时器

2024-04-02 19:04:59 609人浏览 独家记忆
摘要

这篇文章主要讲解了“怎么理解并掌握的Go高级并发模式计时器”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“怎么理解并掌握的Go高级并发模式计时器”吧!前言如果

这篇文章主要讲解了“怎么理解并掌握的Go高级并发模式计时器”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“怎么理解并掌握的Go高级并发模式计时器”吧!

前言

如果你认为结合 Goroutines 去处理时间和计数器很简单的话,那你就错了,这里有提到的一些与 time.Timer 相关的问题或 bug:

  • time: Timer.Reset is not possible to use correctly #14038[1]

  • time: Timer.C can still trigger even after Timer.Reset is called  #11513[2]

  • time: document proper usage of Timer.Stop #14383[3]

看完上面的链接内容后,如果你依然认为很简单,那来看看下面的代码,如下代码会产生死和竞争条件

tm := time.NewTimer(1)tm.Reset(100 * time.Millisecond)<-tm.Cif !tm.Stop() {<-tm.C}

死锁代码片段

func toChanTimed(t *time.Timer, ch chan int) {t.Reset(1 * time.Second)defer func() {if !t.Stop() {<-t.C}}()select {case ch <- 42:case <-t.C:}}

可能代码比较难懂,下面对相关方法进行阐述。

time.Ticker

type Ticker struct {C <-chan Time // The channel on which the ticks are delivered.}

Ticker 简单易用,但也有一些小问题

  • 如果 C 中已存在一条消息,则发送消息时将删除所有未读值。

  • 必须有停止操作:否则 GC 无法回收它

  • 设置 C 无用:消息仍将在原始的 channel 上发送。

time.Tick

time.Tick 是对 time.NewTicker 的封装。最好不要使用该方法,除非你准备将 chan  作为返回结果并在程序的整个生命周期中继续使用它。正如官方描述:

垃圾收集器无法恢复底层的 Ticker,出现 " 泄漏 ". 请谨慎使用,如有疑问请改用 Ticker。

time.After

这与 Tick 的概念基本相同,它是对 Timer 进行封装。一旦计时器被触发,它将被回收。请注意,计时器使用了缓存容量是 1  的通道,即使没有接收者,它仍可以进行计数。如上所述,如果您关心性能且希望能够取消计时,那么你不应该使用 After。

time.Timer ( 也称为 time.WhatTheFork?!)

对于 Go 来说这是一个比较奇怪的 api :NewTicker(Duration) 返回了一个 *Timer 类型,该类型仅暴露一个定义为 chan  类型的变量 C ,这点非常奇怪。

通常在 Go 语言中允许导出的字段意味着用户可以获取或设置该字段,而此处设置变量 C 并没有实际意义。相反:设置 C 并重置 Timer 并不会影响之前在  C 通道的消息传递。更糟糕的是:AfterFunc 返回的 Timer 根本不会使用到 C。

这样看来,Timer 很奇怪,以下是 API 的概述:

type Timer struct {C <-chan Time}func AfterFunc(d Duration, f func()) *Timerfunc NewTimer(d Duration) *Timerfunc (*Timer) Stop(bool)func (*Timer) Reset(d Duration) bool

四个非常简单的函数,其中两个是构造函数,有可能出错吗?

time.AfterFunc

官方文档:AfterFunc 持续时间超时后通过开 Goroutine 去调用 f 函数,返回一个 Timer 类型,以便通过 Stop  方法取消调用。

这么描述虽然没有问题,但需要注意:当调用 Stop 方法时,如果返回 false  ,则表示该函数已经执行且停止失败。但并不意味着函数已经返回,你需要添加一些处理逻辑:

done := make(chan struct{})f := func() {doStuff()close(done)}t := time.AfterFunc(1*time.Second, f)if !t.Stop() {<-done}

这个在 Stop 文档中有相关说明。

除此之外,返回的计时器不会被触发,只能用于调用 Stop 方法。

t := time.AfterFunc(1*time.Second, func() {fmt.Println("Time has passed!")})// This will deadlock.<-t.C

此外,写这篇文章的时候,重置计时器会在传入重置函数的时间段过去后再次调用 f,但这种特性目前暂没有文档规范,未来可能会被改变。

time.NewTimer

官方文档 : NewTimer 实例化 Timer 结构体,在持续时间 d 之后发送当前时间至通道内 .

这意味着没有声明它就无法构建有效的 Timer  类型结构体。如果你需要构建一个以便后续重复使用,可以用该方法进行实例化,或者使用如下代码实现自主创建和停止计数器

t := time.NewTimer(0)if !t.Stop() {<-t.C}

你必须从 channel 中读取数据。假如在 New 和 Stop 调用期间触发了定时器,且 channel 存在未消费的数据, 则 C  会存在一个值。将导致后续读取均是错误的。

(*time.Timer).Stop

Stop 方法会阻止计时器触发。如果调用停止计时器的方法,则返回 true,如果计时器已超时或者已停止,则返回 false。

以上句子中的“或”非常重要。文档中所以关于 Stop 的示例都显示了以下代码片段:

if !t.Stop() {<-t.C}

关键点在于 "or" 它意味着有效 0 次或 1 次。对已消费完通道数据和在此期间未调用 Reset  进行过多次执行的情况,均是无效的。综上所述,当且仅当没有执行对通道数据的消费,Stop+drain 才是安全的。

在文档中体现如下:

例如:假设程序尚未从 t.C 接收数据:

此外,上面的模式不是线程安全的,因为当消费完通道数据时,Stop 返回的值可能已经过时了,两个 Goroutine 尝试消费通道 C  数据也会导致死锁。

(*time.Timer).Reset

这个方法更有意思,文档很长,你可以在这里[4] 进行查看

文档中一个有趣的摘录:

请注意,因为在清空 channel 和计数器到期之间存在竞争条件,我们无法正确使用 Reset 返回值。Reset 方法必须作用于已停止或已过期的  channel 上。

文档所提供 Reset 正确使用方法如下:

if !t.Stop() {<-t.C}t.Reset(d)

不能与来自通道的其他接收者同时使用 Stop 和 Reset 方法, 为了使 C 上传递的消息有效,C 应该在每次 重置 之前被消费完。

重置计时器而不清空它将使运行过程时丢弃该值,因为 C 缓存为 1,运行时对其他执行是有损发送[5]。

time.Timer: 把这些方法放在一起

  • Stop 仅作用在 New 和 Reset 方法之后才安全

  • Reset 仅在 Stop 方法后有效。

  • 只有在每次运行 Stop 后,channel 消费完时,所接收的值才是有效的。

  • 只有 channel 未被消费时,才允许清空 channel。

timer.png

如下是一个正确复用计时器的例子,它解决了文章开头提到的一些问题:

func toChanTimed(t *time.Timer, ch chan int) {t.Reset(1 * time.Second)// No defer, as we don't know which// case will be selectedselect {case ch <- 42:case <-t.C:// C is drained, early returnreturn}// We still need to check the return value// of Stop, because t could have fired// between the send on ch and this line.if !t.Stop() {<-t.C}}

上述代码可以确保 toChanTimed 返回后可以重新使用计时器

感谢各位的阅读,以上就是“怎么理解并掌握的Go高级并发模式计时器”的内容了,经过本文的学习后,相信大家对怎么理解并掌握的Go高级并发模式计时器这一问题有了更深刻的体会,具体使用情况还需要大家实践验证。这里是编程网,小编将为大家推送更多相关知识点的文章,欢迎关注!

--结束END--

本文标题: 怎么理解并掌握的Go高级并发模式计时器

本文链接: https://www.lsjlt.com/news/85249.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • 怎么理解并掌握的Go高级并发模式计时器
    这篇文章主要讲解了“怎么理解并掌握的Go高级并发模式计时器”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“怎么理解并掌握的Go高级并发模式计时器”吧!前言如果...
    99+
    2024-04-02
  • 掌握golang中Select Channels Go并发式编程的高级技巧
    掌握golang中Select Channels的高级技巧可以帮助我们更好地进行并发式编程。下面是一些可以帮助你提升技能的技巧:1....
    99+
    2023-10-20
    Golang
  • go语言的高级并发模式怎么应用
    Go语言的高级并发模式有很多种,可以根据具体的应用场景选择合适的模式。以下是一些常见的高级并发模式及其应用: 扇出-扇入模式(F...
    99+
    2023-10-25
    go语言
  • go语言的高级并发模式怎么实现
    Go语言的高级并发模式可以通过以下几种方式实现: 基于通道的并发模式:Go语言通过通道(Channel)来实现并发的通信和同步。...
    99+
    2023-10-27
    go语言
  • 怎么理解并掌握mysql的表
    本篇内容介绍了“怎么理解并掌握mysql的表”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!一.索引组织表如...
    99+
    2024-04-02
  • 怎么理解并掌握JS装饰器
    本篇内容介绍了“怎么理解并掌握JS装饰器”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!1. 前言装饰器是最...
    99+
    2024-04-02
  • 怎么理解并掌握python正则表达式和re模块
    这篇文章主要介绍“怎么理解并掌握python正则表达式和re模块”,在日常操作中,相信很多人在怎么理解并掌握python正则表达式和re模块问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”怎么理解并掌握pyth...
    99+
    2023-06-01
  • 怎么理解并掌握mysql中的information_schema
    本篇内容介绍了“怎么理解并掌握mysql中的information_schema”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细...
    99+
    2024-04-02
  • 怎么理解并掌握Java的AVL树
    这篇文章主要介绍“怎么理解并掌握Java的AVL树”,在日常操作中,相信很多人在怎么理解并掌握Java的AVL树问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”怎么理解并掌握J...
    99+
    2024-04-02
  • 怎么理解并掌握JavaScript中的this
    这篇文章主要介绍“怎么理解并掌握JavaScript中的this”,在日常操作中,相信很多人在怎么理解并掌握JavaScript中的this问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望...
    99+
    2024-04-02
  • go高并发时append出错怎么解决
    今天小编给大家分享一下go高并发时append出错怎么解决的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。背景在实现图片转码的...
    99+
    2023-07-04
  • 怎么理解并掌握Rust包管理器Cargo
    本篇内容主要讲解“怎么理解并掌握Rust包管理器Cargo”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“怎么理解并掌握Rust包管理器Cargo”吧!Rust 是一种现代编程语言,可提供高性能、...
    99+
    2023-06-16
  • 高效并发处理的Go语言轻量级线程模型
    Go语言的轻量级线程模型与高效并发处理 随着互联网的快速发展,高并发处理已成为现代软件开发的重要需求之一。在传统的多线程编程中,线程的创建和销毁会消耗大量的系统资源,而且线程之间的通信和同步也会带来额外的开销...
    99+
    2024-01-23
    Go语言 轻量级线程模型 高效并发处理
  • 怎么理解并掌握mysql的show processlist time负数
    本篇内容介绍了“怎么理解并掌握mysql的show processlist time负数”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望...
    99+
    2024-04-02
  • 怎么理解并掌握JavaScript中的this关键字
    这篇文章主要讲解了“怎么理解并掌握JavaScript中的this关键字”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“怎么理解并掌握JavaScript中的...
    99+
    2024-04-02
  • 怎么理解高并发下的数据库分布式事务
    这篇文章主要介绍“怎么理解高并发下的数据库分布式事务”,在日常操作中,相信很多人在怎么理解高并发下的数据库分布式事务问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”怎么理解高并发下的数据库分布式事务”的疑惑有所...
    99+
    2023-06-02
  • 怎么提高服务器的并发处理能力
    这篇文章主要介绍怎么提高服务器的并发处理能力,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!什么是服务器并发处理能力一台服务器在单位时间里能处理的请求越多,服务器的能力越高,也就是服务器并发处理能力越强有什么方法衡量服...
    99+
    2023-06-04
  • 怎么理解并掌握web前端性能优化的重排和重绘
    这篇文章主要介绍“怎么理解并掌握web前端性能优化的重排和重绘”,在日常操作中,相信很多人在怎么理解并掌握web前端性能优化的重排和重绘问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大...
    99+
    2024-04-02
  • 怎么理解java高并发的用户线程和守护线程
    这篇文章主要讲解了“怎么理解java高并发的用户线程和守护线程”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“怎么理解java高并发的用户线程和守护线程”吧!守护线程是一种特殊的线程,在后台默...
    99+
    2023-06-25
  • 怎么深入理解Java多线程与并发框中的顺序一致性模型
    怎么深入理解Java多线程与并发框中的顺序一致性模型,相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。一、竞态条件(Race Condition)计算的正确性取决于 多个线程 执行...
    99+
    2023-06-05
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作