go atexit源码分析

文章目录

  • atexit源码解析
    • UML类图
    • 样例一: 程序退出之前执行注册函数
      • 1.1 流程图
      • 1.2 代码分析
    • 样例二:使用cancel取消注册函数
      • 2.1 cancel流程图
      • 2.2 代码分析
    • 样例三:使用Fatal/Fatalln/Fatal执行注册函数
      • 3.1 Fatal/Fatalln/Fatal流程图
      • 3.2 代码分析

atexit源码解析

当我们在执行程序的时候如果想要在退出程序的时候,执行一些清理函数或者日志输出函数等,那么atexit将会是一个很好选择!可以方便地在程序结束之前执行对应的函数。本文将会对go语言中的atexit的源码进行详细的分析,并且给出了对应的使用示例方便大家学习!

源码地址: atexit

UML类图

样例一: 程序退出之前执行注册函数

1.1 流程图

在这里插入图片描述

1.2 代码分析

package mainimport ("fmt""github.com/tebeka/atexit"
)func handler() {fmt.Println("Exiting")
}func main() {atexit.Register(handler)atexit.Exit(0)
}
// output:
// Exiting

首先,我们首先来解析atexit.Register(handler)这个函数都干了些什么:

var (handlersLock  sync.RWMutex  // protects the above twonextHandlerID uint   // 注册函数唯一的标识符handlers      = make(map[HandlerID]func())  // 使用 make 函数创建了一个空的映射。这个初始化会在程序启动时执行,确保了 handlers 在使用前已经被正确地分配和初始化
)
type HandlerID uint 
/****** Register函数 *****/
func Register(handler func()) HandlerID {// 由于handlersLock是一个全局互斥锁 (handlersLock),用于在并发执行中保护nextHandlerID这个全局变量,确保不会发生竞争条件handlersLock.Lock()// 确保函数结束的时候释放锁,避免死锁的情况defer handlersLock.Unlock()// nextHandlerID代表注册函数唯一的标识符,要确保其唯一性,因此要加锁nextHandlerID++// 将id转换为HandlerID类型id := HandlerID(nextHandlerID)// handlers是一个全局映射,键是注册函数的唯一ID,而值为注册函数,该map用于存储所有的注册函数handlers[id] = handler// 返回handler的唯一IDreturn id
}

​ 由此可以得知,当我们使用atexit.Register(handler)之后,就会将handler函数成功注册到全局映射handlers中了,之后就可以通过全局handlers来处理注册函数了。

​ 然后,让我们继续看atexit.Exit(0)做了些什么:

var (...once sync.Once // sync.Once可以确保在并发程序中某个函数只执行一次,无论它被多次调用
)func runHandler(handler func()) {defer func() {// 使用了 recover() 函数来捕获可能的 panic。如果发生 panic,将错误信息输出到标准错误流(os.Stderr),之后会立刻结束这个goroutine,但是不会结束整个程序的运行,这样做可以避免整个程序崩溃.if err := recover(); err != nil {fmt.Fprintln(os.Stderr, "error: atexit handler error:", err)}}()// 调用实际传入的函数,在本例中会直接调用func handler()handler()
}func executeHandlers() {// 使用了读锁(RLock)和读锁解除(RUnlock),用于在并发执行中保护对全局变量 handlers 的读取操作。handlersLock.RLock()defer handlersLock.RUnlock()// 读取已经注册过的所有handler并且执行它们,这个操作是并发安全的for _, handler := range handlers {runHandler(handler)}
}func runHandlers() {// 无论runHandlers函数被调用多少次,在同一个程序运行周期内,executeHandlers函数只会被执行一次。// 确保当并发调用runHandlers函数的时候,所有的注册函数只会执行一次once.Do(executeHandlers)
}/****** Exit函数 *****/
// Exit runs all the atexit handlers and then terminates the program using
// os.Exit(code)
func Exit(code int) {runHandlers()os.Exit(code)
}

​ 由此可知,当我们调用atexit.Exit(0)的时候,程序会先执行所有的注册函数,之后才会调用os.Exit退出整个程序。

样例二:使用cancel取消注册函数

2.1 cancel流程图

在这里插入图片描述

2.2 代码分析

当我们不想执行注册函数的时候,可是函数又已经注册了,那么就可以使用cancel取消注册函数的执行。

package mainimport ("fmt""github.com/tebeka/atexit"
)func handler() {fmt.Println("handler Exiting")
}
func handler2() {fmt.Println("handler2 Exiting")
}
func main() {id := atexit.Register(handler)err := id.Cancel()if err != nil {fmt.Println("Error cancel")return}atexit.Register(handler2)atexit.Exit(0)
}
// Output:
// handler2 Exiting

​ 让我们看看 err := id.Cancel()干了些什么:

// Cancel cancels the handler associated with id
func (id HandlerID) Cancel() error {handlersLock.Lock()defer handlersLock.Unlock()// 检查是否存在对应id的注册函数。如果handlers中不存在该 id,则返回一个包含错误信息的 error 类型。也就是说只能够取消已经注册函数,不能够取消未被注册的函数。_, ok := handlers[id]if !ok {return fmt.Errorf("handler %d not found", id)}// 删除handlers对应id的handler注册函数delete(handlers, id)return nil
}

​ 可以看到,其实cancel函数也只是把对应id的注册函数从hanlders中移除而已,之后执行注册函数的时候就不会执行该函数了。

样例三:使用Fatal/Fatalln/Fatal执行注册函数

3.1 Fatal/Fatalln/Fatal流程图

在这里插入图片描述

3.2 代码分析

package mainimport ("fmt""github.com/tebeka/atexit"
)func handler() {fmt.Println("Exiting")
}func main() {atexit.Register(handler)// 以下三个语句只能够执行一个其中一个,因为执行完对应的语句就会退出程序。// atexit.Fatal("this is a Fatal message")// atexit.Fatalf("this is a Fatalf message:%s", "fatalf")atexit.Fatalln("this is a Fatalln message")
}

​ 当执行atexit.Fatal或者atexit.Fatalf或者atexit.Fatalln会首先执行注册函数,之后才会执行对应的Fatal/Fatalf/Fatalln函数来退出程序。

​ 让我们看看这三个函数都干了些什么:

// Fatal runs all the atexit handler then calls log.Fatal (which will terminate
// the program)
func Fatal(v ...interface{}) {runHandlers()log.Fatal(v...)
}// Fatalf runs all the atexit handler then calls log.Fatalf (which will
// terminate the program)
func Fatalf(format string, v ...interface{}) {runHandlers()log.Fatalf(format, v...)
}// Fatalln runs all the atexit handler then calls log.Fatalln (which will
// terminate the program)
func Fatalln(v ...interface{}) {runHandlers()log.Fatalln(v...)
}

​ 可以看到,它们首先通过runHandlers()调用了所有的已经注册函数,之后再通过log.Fatal/Fatalf/Fatalln来输出日志,并且最终退出程序。
注:log.Fatal/Fatalf/Fatalln在输出日志之后会自动终止程序的运行。

到此,atexit源码分析就结束了!若文章中出现任何纰漏,欢迎大家指正批评哦!如果觉得写得还不错的话,麻烦大家点赞收藏加关注哦!

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

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

相关文章

Android平台GB28181设备接入模块开发填坑指南

技术背景 为什么要开发Android平台GB28181设备接入模块?这个问题不再赘述,在做Android平台GB28181客户端的时候,媒体数据这块,我们已经有了很好的积累,因为在此之前,我们就开发了非常成熟的RTMP推送、轻量…

Scannet v2 数据集介绍以及子集下载展示

Scannet v2 数据集介绍以及子集下载展示 文章目录 Scannet v2 数据集介绍以及子集下载展示参考数据集简介子集scannet_frames_25kscannet_frames_test 下载脚本 download_scannetv2.py 参考 scannet数据集简介和下载-CSDN博客 scannet v2 数据集下载_scannetv2数据集_蓝羽飞鸟的…

BeanUtil的正确使用方式

shigen日更文章的博客写手,擅长Java、python、vue、shell等编程语言和各种应用程序、脚本的开发。记录成长,分享认知,留住感动。 在实际的开发中,我们常常会用到工具类去拷贝对象的属性,将一个对象的属性转换成另外一个…

球面的表面积

此推导需要用到重积分的知识,另外关于曲面的面积公式可以看我之前的博客

百度AI布局:从财报看百度的核心竞争力和未来发展方向

百度是中国最大的搜索引擎,也是全球领先的人工智能(AI)公司。百度在2023年第三季度业绩中,展示了其在AI领域的强劲表现和广阔前景。 百度财报透露了关于AI业务的哪些重要信息? 百度在2023年第三季度的财报中&#xf…

机器学习:攻击方法FGSM系列

任务 FGSM I-FGSM MI-FGSM Ensemble Attack 攻击评价指标 准确率越低表明攻击越好 数据 预训练模型 BaseLine 实践

***Linux常用命令及解释

1、查看Linux的版本信息 1.1、uname -a 1.2、cat /etc/issue 1.3、cat /proc/version 1.4、hostnamectl 通过使用hostnamectl命令,可以查询和更改系统主机名,并且还可以查看Linux的发行版和内核版本。 2、删除文件 3、修改目录权限 4、解压文件 5、…

探索 Linux vim/vi 编辑器:介绍、模式以及基本操作演示

💐作者:insist-- 💐个人主页:insist-- 的个人主页 理想主义的花,最终会盛开在浪漫主义的土壤里,我们的热情永远不会熄灭,在现实平凡中,我们终将上岸,阳光万里 ❤️欢迎点…

Spring的依赖注入,依赖注入的基本原则,依赖注入的优势

文章目录 Spring的依赖注入依赖注入的基本原则依赖注入有什么优势查找定位操作与应用代码完全无关。有哪些不同类型的依赖注入实现方式?构造器依赖注入和 Setter方法注入的区别 Spring的依赖注入 控制反转IoC是一个很大的概念,可以用不同的方式来实现。…

android系统新特性——用户界面以及系统界面改进

用户界面改进 Android用户界面改进最明显的就是MD了。MD是Google于2014年推出的设计语言,它是一套完整的设计系统,包含了动画、样式、布局、组件等一系列与设计有关的元素。通过对这些行为的描述,让开发者设计出更符合目标的软件&#xff0c…

vue3+elementPlus之侧边菜单栏功能

选择默认的颜色&#xff0c;将代码拷贝至<el-aside>模块中 稍微把不需要的修改一下。 <template><div class"common-layout"><el-container><el-header class"homeHeader"><div class"headerTitle">Devops…

vue怎么实现国际化? vue-i18n 插件实现国际化,支持切换不同语言

依赖的文档开始 | Vue I18n 一、安装 npm install vue-i18n 如果在一个模块系统中使用它&#xff0c;你必须通过 Vue.use() 明确地安装 vue-i18n&#xff1a; import Vue from vue import VueI18n from vue-i18nVue.use(VueI18n)二、使用 在 src 下创建 lang 文件夹 1.准…

实现【Linux--NTP 时间同步服务搭建】

实现【Linux--NTP 时间同步服务搭建】 &#x1f53b; 前言&#x1f53b; 一、NTP 校时&#x1f530; 1.1 NTP 服务校时与 ntpdate 校时的区别&#x1f530; 1.2 NTP 校时服务搭建&#x1f530; 1.2.1 确认 ntp 的安装&#x1f530; 1.2.2 配置 ntp 服务&#x1f530; 1.2.3 启动…

大数据面试大厂真题【附答案详细解析】

1.Java基础篇&#xff08;阿里、蚂蚁、字节、携程、快手、杭州银行等&#xff09; 问题&#xff1a;HashMap的底层实现原理 答案&#xff1a; 在jdk1.8之前&#xff0c;hashmap由 数组-链表数据结构组成&#xff0c;在jdk1.8之后hashmap由 数组-链表-红黑树数据结构组成&…

openpnp - 给底部相机加防尘罩

文章目录 openpnp - 给底部相机加防尘罩概述笔记END openpnp - 给底部相机加防尘罩 概述 设备标定完, 看着底部相机, 有点担心掉进去东西, 万一从吸嘴掉下去的料(或者清理设备台面时, 不小心掉进去东西)将顶部相机搞短路怎么办. 就想加个防尘罩, 如果有东西掉进去, 可以掉到机…

makefile 学习(5)完整的makefile模板

参考自&#xff1a; (1&#xff09;深度学习部署笔记(二): g, makefile语法&#xff0c;makefile自己的CUDA编程模板(2&#xff09;https://zhuanlan.zhihu.com/p/396448133(3) 一个挺好的工程模板&#xff0c;(https://github.com/shouxieai/cpp-proj-template) 1. c 编译流…

开源vs闭源,处在大模型洪流中,向何处去?

文章目录 一、开源和闭源的优劣势比较1.1 开源优势1.2 闭源的优势 二、开源和闭源对大模型技术发展的影响2.1 数据共享2.2 算法创新2.3 业务拓展2.4 安全性和隐私2.5 社会责任和伦理 三、开源与闭源的商业模式比较3.1 盈利模式3.2 市场竞争3.3 用户生态3.4 创新速度 四&#xf…

shiro的前后端分离模式

shiro的前后端分离模式 前言&#xff1a;在上一篇《shiro的简单认证和授权》中介绍了shiro的搭建&#xff0c;默认情况下&#xff0c;shiro是通过设置cookie&#xff0c;使前端请求带有“JSESSION”cookie&#xff0c;后端通过获取该cookie判断用户是否登录以及授权。但是在前…

TCP/IP协议、三次握手、四次挥手

TCP/IP TCP/IP协议分层TCP头部三次握手TCP四次挥手常见问题1、什么是TCP网络分层2、TCP为什么是三次握手&#xff0c;不是两次或者四次&#xff1f;3、TCP为什么是四次挥手&#xff0c;为什么不能是三次挥手将第二次挥手和第三次挥手合并&#xff1f;4、四次挥手时为什么TIME_W…

ZKP11.4 Use CI to instantiate Fiat-Shamir

ZKP学习笔记 ZK-Learning MOOC课程笔记 Lecture 11: From Practice to Theory (Guest Lecturer: Alex Lombardi) 11.4 Use CI to instantiate Fiat-Shamir Avoid Bad Challenges Def: Given false claim x x x and a first message α \alpha α, a challenge β \beta …