go atexit源码分析

news2024/11/28 8:27:21

文章目录

  • 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 main

import (
	"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 two
    nextHandlerID 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的唯一ID
	return 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 main

import (
	"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 main

import (
	"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.coloradmin.cn/o/1254376.html

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

相关文章

什么是LASSO回归,怎么看懂LASSO回归的结果

随着机器学习的发展,越来越多SCI文章都使用了更多有趣、高效的统计方法来进行分析,LASSO回归就是其中之一。很多小伙伴听说过LASSO,但是对于LASSO是什么,有什么用,怎么才能实现,大家可能一头雾水。今天的文…

每日一题2023.11.26——个位数统计【PTA】

题目要求: 输入格式: 每个输入包含 1 个测试用例,即一个不超过 1000 位的正整数 N。 输出格式: 对 N 中每一种不同的个位数字,以 D:M 的格式在一行中输出该位数字 D 及其在 N 中出现的次数 M。要求按 D 的升序输出。…

【华为数通HCIP | 网络工程师】821-IGP高频题、易错题之OSPF(7)

个人名片: 🐼作者简介:一名大三在校生,喜欢AI编程🎋 🐻‍❄️个人主页🥇:落798. 🐼个人WeChat:hmmwx53 🕊️系列专栏:🖼️…

blender 3D眼球结构

角膜(Cornea):眼球的前部,透明的曲面,负责折射光线。虹膜(Iris):眼睛的颜色部分,控制瞳孔大小以调整进入眼睛的光量。瞳孔(Pupil):虹膜…

S25FL系列FLASH读写的FPGA实现

文章目录 实现思路具体实现子模块实现top模块 测试Something 实现思路 建议读者先对 S25FL-S 系列 FLASH 进行了解,我之前的博文中有详细介绍。 笔者的芯片具体型号为 S25FL256SAGNFI00,存储容量 256Mb,增强高性能 EHPLC,4KB 与 6…

快速幂算法详解(C++实现)

文章目录 1. 什么是快速幂2. 暴力求解代码实现缺陷分析 3. 优化一:取模运算的性质4. 优化二:快速幂算法的核心思想5. 终极优化:位运算优化6. 源码 这篇文章我们来一起学习一个算法——快速幂算法。 1. 什么是快速幂 顾名思义,快速…

中海油“海安杯”一站到底知识竞赛真的很有特色

中海油“海安杯”一站到底知识竞赛规格高,赛制复杂,天纵知识竞赛系统为此次知识竞赛提供了软件支持。本次竞赛设置选手区和擂台区两个区域。比赛共分为五个轮次,五个轮次选手区所有参赛选手均需答题。 第一轮:“脱颖而出” 所有参…

叠加原理(superposition principle)

叠加原理(superposition principle)指对线性系统而言,两个或多个输入产生的输出,等于这几个输入单独引起的输出的和,即输入的叠加等于各输入单独引起的输出的叠加。 例如,如果输入产生的输出是,…

B树与B+树的对比

B树: m阶B树的核心特性: 树中每个节点至多有m棵子树,即至多含有m-1个关键字根节点的子树数属于[2, m],关键字数属于[1, m-1],其他节点的子树数属于 [ ⌈ m 2 ⌉ , m ] [\lceil \frac{m}{2}\rceil, m] [⌈2m​⌉,m]&am…

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

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

电子学会C/C++编程等级考试2022年09月(二级)真题解析

C/C++等级考试(1~8级)全部真题・点这里 第1题:统计误差范围内的数 统计一个整数序列中与指定数字m误差范围小于等于X的数的个数。 时间限制:5000 内存限制:65536输入 输入包含三行: 第一行为N,表示整数序列的长度(N <= 100); 第二行为N个整数,整数之间以一个空格分…

【教学类-06-12】20231126 (一)如何让加减乘除题目从小到大排序(以1-20之间加法为例,做正序排列用)

结果展示 优化后 优化前 背景需求&#xff1a; 生成列表 单独抽取显示题目排序方法 存在问题: 我希望 00 01 02……这样排序&#xff0c;但是实际上&#xff0c;除了第一个加数会从小到大排序&#xff0c;第二个被加数的第十位数和个位数都会从小到大排序&#xff0c;也就是…

定长子网划分和变长子网划分问题_二叉树解法_通俗易懂_配考研真题

引入:定长子网划分和变长子网划分的基本概念 定长子网划分和变长子网划分的基本概念 目前常用的子网划分&#xff0c;是基于CIDR的子网划分&#xff0c;也就是将给定的CIDR地址块划分为若干个较小的CIDR地址块。 定长子网划分: 使用同一个子网掩码来划分子网&#xff0c;因…

使用VC++设计程序对一幅256级灰度图像进行全局固定阈值分割、自适应阈值分割

图像分割–全局固定阈值分割、自适应阈值分割 获取源工程可访问gitee可在此工程的基础上进行学习。 该工程的其他文章&#xff1a; 01- 一元熵值、二维熵值 02- 图像平移变换&#xff0c;图像缩放、图像裁剪、图像对角线镜像以及图像的旋转 03-邻域平均平滑算法、中值滤波算法、…

线性表,也是Java中数组的知识点!

线性表定义&#xff1a; 由n (n≥0)个数据特性相同的元素构成的有限序列称为线性表&#xff0c;(n0)的时候被称为空表。 线性表的顺序表示 线性表的顺序存储又被称为顺序表 优点 无需为表示表中元素之间的逻辑关系而增加额外的存储空间可以随意读取任意位置的元素 缺点 插入…

Autosar MCAL-RH850P1HC-MCAL配置环境搭建

文章目录 前言下载安装包软件安装安装SIP包安装MCAL文件配置工程配置生成代码测试静态代码路径总结前言 对于RH850P1HC,官网有免费的MCAL,但官网的MCAL没有CAN模块(原厂反馈为Bosch IP,CAN Driver他们没有),也没有FEE模块。如果需要,可以找第三方软件公司,如ETAS.虽然M…

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

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

SpringBoot——LiteFlow引擎框架

优质博文&#xff1a;IT-BLOG-CN 一、LiteFlow 简介 LiteFlow是一个轻量且强大的国产规则引擎框架&#xff0c;可用于复杂的组件化业务的编排领域。帮助系统变得更加丝滑且灵活。利用LiteFlow&#xff0c;你可以将瀑布流式的代码&#xff0c;转变成以组件为核心概念的代码结构…

计算机组成原理-Cache的基本概念和原理

文章目录 存储系统存在的问题Cache的工作原理局部性原理性能分析例题界定何为局部部分问题总结 存储系统存在的问题 增加Cache层来缓和CPU和主存的工作速度矛盾 Cache的工作原理 启动某个程序后&#xff0c;将程序的代码从辅存中取出放入内存中&#xff0c;再从内存中将代码…

nginx反向代理解决跨域前端实践

需求实现 本地请求百度的一个搜索接口&#xff0c;用nginx代理解决跨域思路&#xff1a;前端和后端都用nginx代理到同一个地址8080&#xff0c;这样访问接口就不存在跨域限制 本地页面 查询一个百度搜索接口&#xff0c;运行在http://localhost:8035 index.js const path …