逐步学习Go-sync.Once(只执行一次)Exactly Once

sync.Once简介

sync.Once 是一个会执行且仅执行一次动作的对象。该对象在第一次使用后不能再被复制。

在 Go 内存模型的术语中,sync.Once 的 Do 方法中的函数 f 返回的操作,相对于任何对 once.Do(f) 的调用返回的操作,都具有“同步优先”的顺序。简单来说,即使在并发环境下,函数 f 也只会在首次调用 once.Do(f) 时执行。

每个 sync.Once 对象仅适用于执行一次动作。也就是说,如果多次调用了 once.Do(f),仅第一次的调用会激发函数 f 的执行,即使每次调用 once.Do(f) 时函数 f 的值都有所不同。

sync.Once 一般用于必须仅初始化一次的场景。由于作为 Do 方法参数的函数没有传入的参数,如果你需要在由 Do 方法调用的函数中使用特定的参数,你可能需要使用闭包:

config.once.Do(func() { config.init(filename) })

COPY

由于 Do 方法在其内部的函数返回之前不会返回,如果函数 f 导致 Do 方法被调用,就会引发死锁。

如果 Do 方法的函数 f 发生 panic,sync.Once 会认为函数 f 已执行完毕。以后再调用 Do 方法时,不会再执行函数 f

功能特性测试

sync.Once还是比较简单的,而且源代码也特别简单,我们来列举几个场景测试。
以下是每个测试用例所覆盖的特性和场景的描述:

  1. TestOnce_ShouldExecuteOnce_WhenExecuteOnlyOnce:这个测试用例验证了sync.Once的基本特性 – 即确保一段代码在非并发环境中只执行一次。

  2. TestOnce_ShouldExecuteOnce_WhenExecutedOnceAndExecuteAgain:这个测试用例验证了即使sync.OnceDo方法被多次调用,内部的函数也只执行一次。这是sync.Once的核心特性,用于确保一个操作在整个程序运行期间只执行一次,无论这个操作被尝试执行多少次。

  3. TestOnce_ShouldNotExecute_WhenFunctionPanicked:这个测试用例验证了当sync.OnceDo方法的函数执行过程中发生panic时,随后的Do调用将不会继续执行函数。这是sync.Once的一个重要特性,它确保了即使在面临错误处理的情况下,被追踪的函数只执行一次。

  4. TestOnce_ShouldExecuteAgain_WhenPreviousExecutionPaniced:这个测试用例验证了即使Do方法的函数因为panic而没有正常执行,sync.Once也会认为该函数已经执行过,并且不会在后续的Do方法调用中再次尝试执行函数。这是sync.Once的另一个重要特性,它可以帮助规避错误以及异常的发生。

  5. TestOnce_ShouldExecuteOnce_WhenCalledInMultipleGoroutines:这个测试用例验证了即使在多个goroutine并发调用sync.OnceDo方法时,函数也只会执行一次。这是sync.Once被设计用来解决的主要问题,也是sync.Once在并发编程中的一个重要应用场景。

测试代码


import ("sync""sync/atomic""testing""github.com/stretchr/testify/assert"
)func TestOnce_ShouldExecuteOnce_WhenExecuteOnlyOnce(t *testing.T) {var i intonce := sync.Once{}once.Do(func() {i = 1000})assert.Equal(t, 1000, i)
}func TestOnce_ShouldExecuteOnce_WhenExecutedOnceAndExecuteAgain(t *testing.T) {var i intonce := sync.Once{}once.Do(func() {i = 1000})once.Do(func() {i = 2000})assert.Equal(t, 1000, i)
}func TestOnce_ShouldNotExecute_WhenFunctionPanicked(t *testing.T) {var num intonce := sync.Once{}// 第一次调用应该 panicassert.Panics(t, func() {once.Do(func() {panic("Error")})})// 因为第一次panic了,此时再调用 Do 方法,函数 f 不应该被执行once.Do(func() {num = 1000})// 因为 num 的值没有被改变,所以应该还是 0assert.Equal(t, num, 0)
}func TestOnce_ShouldExecuteAgain_WhenPreviousExecutionPaniced(t *testing.T) {var num int = 0once := sync.Once{}// 第一次调用应该 panicassert.Panics(t, func() {once.Do(func() {panic("Error")})})// 第一次 panic 后,下一次调用应该正确执行once.Do(func() {num = 1000})assert.Equal(t, 0, num)
}func TestOnce_ShouldExecuteOnce_WhenCalledInMultipleGoroutines(t *testing.T) {var num atomic.Int32once := sync.Once{}wg := sync.WaitGroup{}wg.Add(100)// 启动 100 个 goroutine, 都尝试执行once.Dofor i := 0; i < 100; i++ {go func() {once.Do(func() {num.Add(1)})wg.Done()}()}wg.Wait()assert.Equal(t, int32(1), num.Load())
}

COPY

sync.Once源码


type Once struct {// 用来标识操作是否已经执行过done atomic.Uint32 // 用来在多个并发的`Do`调用中,保证只有一个可以执行函数`f`m    Mutex         
}func (o *Once) Do(f func()) {if o.done.Load() == 0 {// 如果操作还没执行过,进入doSlow方法o.doSlow(f)}
}func (o *Once) doSlow(f func()) {o.m.Lock()defer o.m.Unlock()if o.done.Load() == 0 {// 确保无论f是否panic,done都会被设置为1defer o.done.Store(1) // 执行用户传入的函数ff()                   }
}

COPY

sync.Once能用来干啥?

  1. 单例模式
  2. 延迟初始化
  3. 并发控制

原文链接

逐步学习Go-sync.Once(只执行一次)Exactly Once – FOF编程网

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/794745.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Jupyter Notebook安装使用(一)

1. 简介 Jupyter Notebook 是一个非常强大的工具&#xff0c;它允许用户创建和共享包含实时代码、方程式、可视化和叙事文本的文档。这种工具特别适合数据清理和转换、数值模拟、统计建模、数据可视化、机器学习等多种应用领域。 2. 安装Jupyter Notebook 2.1. 使用 Anaconda…

Java面试题:请解释Java中的多线程编程?

Java中的多线程编程允许 concurrently 执行多个线程&#xff0c;从而可以同时执行多个任务&#xff0c;提高程序的效率和响应性。在Java中&#xff0c;线程可以通过以下两种主要方式来实现&#xff1a; 继承 Thread 类实现 Runnable 接口 继承 Thread 类 class MyThread ext…

Gson的用法

1. 导入依赖 <dependency><groupId>com.google.code.gson</groupId><artifactId>gson</artifactId><version>2.8.6</version> </dependency> 2. 使用Gson进行解析 2.1 Gson解析普通对象 package com.jiang.partnetbackend.…

利用Idea实现Ajax登录(maven工程)

一、新建一个maven工程&#xff08;不会建的小伙伴可以参考Idea引入maven工程依赖(保姆级)-CSDN博客&#xff09;&#xff0c;工程目录如图 ​​​​​​​ js文件可以上up网盘提取 链接&#xff1a;https://pan.baidu.com/s/1yOFtiZBWGJY64fa2tM9CYg?pwd5555 提取码&…

Chatgpt掘金之旅—有爱AI商业实战篇|专业博客|(六)

演示站点&#xff1a; https://ai.uaai.cn 对话模块 官方论坛&#xff1a; www.jingyuai.com 京娱AI 一、AI技术创业博客领域有哪些机会&#xff1f; 人工智能&#xff08;AI&#xff09;技术作为当今科技创新的前沿领域&#xff0c;为创业者提供了广阔的机会和挑战。随着AI技…

ABBYY FineReader15免费电脑OCR图片文字识别软件

产品介绍&#xff1a;ABBYY FineReader 15 OCR图片文字识别软件 ABBYY FineReader 15是一款光学字符识别&#xff08;OCR&#xff09;软件&#xff0c;专门设计用于将扫描的文档、图像和照片中的文本转换成可编辑和可搜索的格式。这款软件利用先进的OCR技术&#xff0c;能够识别…

递归算法解读

递归&#xff08;Recursion&#xff09;是计算机科学中的一个重要概念&#xff0c;它指的是一个函数&#xff08;或过程&#xff09;在其定义中直接或间接地调用自身。递归函数通过把问题分解为更小的相似子问题来解决原问题&#xff0c;这些更小的子问题也使用相同的解决方案&…

微信小程序短链接工具推荐

现在微信小程序大行其道&#xff0c;但工作中大部分人选择了短链接的方式来推广微信小程序&#xff0c;那么微信小程序短链接工具哪个好?今天就分享一篇从网上看到的关于《微信小程序短链接工具推荐》文&#xff0c;作者是souki&#xff0c;一起来看看吧! 一、缩链 1、生成方…

MongoDB聚合运算符:$maxN(用于数组)

文章目录 语法使用举例 $maxN聚合运算符返回数组中最大的n个值。 语法 { $maxN: { n: <expression>, input: <expression> } }参数说明&#xff1a; n&#xff1a;正整数表达式&#xff0c;用于指定返回数组元素的数量。input&#xff1a;可以解析为数组的表达式…

Memcached 教程之 PHP 连接 Memcached 服务(十)

PHP 连接 Memcached 服务 在前面章节中我们已经介绍了如何安装 Memcached 服务&#xff0c;接下来我们为大家介绍 PHP 如何使用 Memcached 服务。 PHP Memcache 扩展安装 PHP Memcache 扩展包下载地址&#xff1a;PECL :: Package :: memcache&#xff0c;你可以下载最新稳定…

【VASP学习】在Ubuntu系统安装vasp.5.4.4的全过程(包括VASP官方学习资料、安装过程中相关编辑器的配置、VASP的编译及VASP的测试)

在Ubuntu系统安装vasp.5.4.4的全过程 VASP的简介与相关学习资料安装前的准备工作及说明安装过程intel编译器的安装VASP的编译VASP的测试 参考来源 VASP的简介与相关学习资料 VASP(Vienna Ab initio Simulation Package)是基于第一性原理对原子尺度的材料进行模拟计算的软件。比…

【Linux】线程概念及线程互斥

目录 线程概念 线程优点 线程缺点 线程异常 线程系统编程接口 线程创建及终止 线程等待 使用线程系统接口封装一个小型的C线程库并实现一个抢票逻辑 线程互斥 互斥量的接口 线程互斥实现原理 使用系统加锁接口封装LockGuard 实现自动化加锁 线程安全和可重入函数 …

css:img引入svg后修改颜色

<img class"group-icon" :class"group.opened ? group-icon-active : ":src"require(/assets/icon/ group.icon .svg)" /> // 默认样式 .group-icon {width: 16px;height: 16px;position: relative;left: -1000px;filter: drop-shadow…

HTML1:html基础

HTML 冯诺依曼体系结构 运算器 控制器 存储器 输入设备 输出设备 c/s(client客户端) 客户端架构软件 需要安装,更新麻烦,不跨平台 b/s(browser浏览器) 网页架构软件 无需安装,无需更新,可跨平台 浏览器 浏览器内核: 处理浏览器得到的各种资源 网页: 结构 HTML(超…

【Rust】生命周期

Rust 生命周期机制是与所有权机制同等重要的资源管理机制。 之所以引入这个概念主要是应对复杂类型系统中资源管理的问题。 引用是对待复杂类型时必不可少的机制&#xff0c;毕竟复杂类型的数据不能被处理器轻易地复制和计算。 但引用往往导致极其复杂的资源管理问题&#x…

Elasticsearch:我们如何演化处理二进制文档格式

作者&#xff1a;来自 Elastic Sean Story 从二进制文件中提取内容是一个常见的用例。一些 PDF 文件可能非常庞大 — 考虑到几 GB 甚至更多。Elastic 在处理此类文档方面已经取得了长足的进步&#xff0c;今天&#xff0c;我们很高兴地介绍我们的新工具 —— 数据提取服务&…

两个数组的dp---区间dp

1、最长公共子序列 1143. 最长公共子序列 - 力扣&#xff08;LeetCode&#xff09; class Solution {//1、状态表示&#xff1a;// 经验 题目要求// (1)选取第一个字符串[0,i]区间&#xff0c;以及第二个字符串[0,j]区间&#xff0c;作为研究对象// (2)根据题目要求&…

SCI一区 | Matlab实现NGO-TCN-BiGRU-Attention北方苍鹰算法优化时间卷积双向门控循环单元融合注意力机制多变量时间序列预测

SCI一区 | Matlab实现NGO-TCN-BiGRU-Attention北方苍鹰算法优化时间卷积双向门控循环单元融合注意力机制多变量时间序列预测 目录 SCI一区 | Matlab实现NGO-TCN-BiGRU-Attention北方苍鹰算法优化时间卷积双向门控循环单元融合注意力机制多变量时间序列预测预测效果基本介绍模型…

解决JavaWeb中IDEA2023新版本无法创建Servlet的问题

出现问题&#xff1a;IDEA右键创建Servlet时&#xff0c;找不到选项 原因分析&#xff1a;IDEA的2023版的已经不支持Servlet了&#xff0c;如果还要使用的话&#xff0c;需要自己创建模板使用 创建模板 右击设置&#xff0c;选择&#xff08;File and Code Templates&#x…

前端基础技术栈教程--HTML、CSS与javascript

在我们身处的这个信息化时代&#xff0c;互联网已成为生活不可或缺的一部分。每当我们打开网站浏览信息时&#xff0c;都离不开三个构建网页的基石&#xff1a;HTML、CSS和JavaScript。这些技术不仅塑造了网页的骨架和外观&#xff0c;更为我们提供了丰富的互动体验。对于渴望掌…