1.如何处理好Golang中的源码panic与recover
2.翻译:XDA论坛教程:如何手动编译TWRP Recovery
3.如何刷手机第三方中文TWRP的recovery
4.Golang源码剖析panic与recover,看不懂你打我好了
5.Gin源码分析 - 中间件(5)- Recovery
如何处理好Golang中的源码panic与recover
Go 语言以其高性能和高并发特性而闻名,特别是源码其提供的 pile -N -l -S main.go就可以看到对应的汇编码了,我们截取部分片段分析:
上面重点部分就是源码画红线的三处,第一步调用runtime.deferprocStack创建defer对象,源码这一步大家可能会有疑惑,源码在线考试系统。net源码我上一文忘记讲个这个了,源码这里先简单概括一下,源码defer总共有三种模型,源码编译一个函数里只会有一种defer模式。源码在讲defer实现机制时,源码我们一起看过defer的源码结构,其中有一个字段就是源码_panic,是源码触发defer的作用,我们来看看的源码panic的结构:
简单介绍一下上面的字段:
上面的pc、sp、goexit我们单独讲一下,runtime包中有一个Goexit方法,提取app源码工具Goext能够终止调用它的goroutine,其他的goroutine是不受影响的,goexit也会在终止goroutine之前运行所有延迟调用函数,Goexit不是一个panic,所以这些延迟函数中的任何recover调用都将返回nil。如果我们在主函数中调用了Goexit会终止该goroutine但不会返回func main。由于func main没有返回,因此程序将继续执行其他gorountine,直到所有其他goroutine退出,程序才会crash。
下面就开始我们的重点吧~。
在讲defer实现机制时,我们一起看过defer的结构,其中有一个字段就是_panic,是触发defer的作用,我们来看看的panic的结构:简单介绍一下上面的字段:上面的pc、sp、充值系统源码搭建goexit我们单独讲一下,runtime包中有一个Goexit方法,Goext能够终止调用它的goroutine,其他的goroutine是不受影响的,goexit也会在终止goroutine之前运行所有延迟调用函数,Goexit不是一个panic,所以这些延迟函数中的任何recover调用都将返回nil。如果我们在主函数中调用了Goexit会终止该goroutine但不会返回func main。由于func main没有返回,因此程序将继续执行其他gorountine,直到所有其他goroutine退出,程序才会crash。写个简单的例子:运行上面的例子你就会发现,即使在主goroutine中调用了runtime.Goexit,其他goroutine是没有任何影响的。所以结构中的pc、sp、决策之王指标源码goexit三个字段都是为了修复runtime.Goexit,这三个字段就是为了保证该函数的一定会生效,因为如果在defer中发生panic,那么goexit函数就会被取消,所以才有了这三个字段做保护。看这个例子:
英语好的可以看一看这个: github.com/golang/go/is...,这就是上面的一个例子,这里就不过多解释了,了解就好。
接下来我们再来看一看gopanic方法。
gopanic的代码有点长,我们一点一点来分析:
根据不同的类型判断当前发生panic错误,这里没什么多说的,接着往下看。
上面的代码都是截段,这些部分都是为了判断当前defer是否可以使用开发编码模式,具体怎么操作的码客跑腿源码就不展开了。
在第三部分进行defer内联优化选择时会执行调用延迟函数(reflectcall就是这个作用),也就是会调用runtime.gorecover把recoverd = true,具体这个函数的操作留在下面讲,因为runtime.gorecover函数并不包含恢复程序的逻辑,程序的恢复是在gopanic中执行的。先看一下代码:
这段代码有点长,主要就是分为两部分:
第一部分主要是这个判断if gp._panic != nil && gp._panic.goexit && gp._panic.aborted { ... },正常recover是会绕过Goexit的,所以为了解决这个,添加了这个判断,这样就可以保证Goexit也会被recover住,这里是通过从runtime._panic中取出了程序计数器pc和栈指针sp并且调用runtime.recovery函数触发goroutine的调度,调度之前会准备好 sp、pc 以及函数的返回值。
第二部分主要是做panic的recover,这也与上面的流程基本差不多,他是从runtime._defer中取出了程序计数器pc和栈指针sp并调用recovery函数触发Goroutine,跳转到recovery函数是通过runtime.call进行的,我们看一下其源码(src/runtime/asm_amd.s 行):
因为go语言中的runtime环境是有自己的堆栈和goroutine,recovery函数也是在runtime环境执行的,所以要调度到m->g0来执行recovery函数,我们在看一下recovery函数:
在recovery 函数中,利用 g 中的两个状态码回溯栈指针 sp 并恢复程序计数器 pc 到调度器中,并调用 gogo 重新调度 g , goroutine 继续执行,recovery在调度过程中会将函数的返回值设置为1。这个有什么作用呢? 在deferproc函数中找到了答案:
当延迟函数中recover了一个panic时,就会返回1,当 runtime.deferproc 函数的返回值是 1 时,编译器生成的代码会直接跳转到调用方函数返回之前并执行 runtime.deferreturn,跳转到runtime.deferturn函数之后,程序就已经从panic恢复了正常的逻辑。
在这里runtime.fatalpanic实现了无法被恢复的程序崩溃,它在中止程序之前会通过 runtime.printpanics 打印出全部的 panic 消息以及调用时传入的参数。
这就是这个逻辑流程,累死我了。。。。
结尾给大家发一个小福利,哈哈,这个福利就是如果避免出现panic,要注意这些:这几个是比较典型的,还有很多会发生panic的地方,交给你们自行学习吧~。
好啦,这篇文章就到这里啦,素质三连(分享、点赞、在看)都是笔者持续创作更多优质内容的动力!
Gin源码分析 - 中间件(5)- Recovery
Recovery中间件在HTTP请求处理中扮演着关键角色,尤其在处理过程中产生panic时。它能够捕获并处理这些异常,确保服务的稳定性和客户端的正常响应。通过使用gin框架,可以通过两种方式集成Recovery中间件:第一种是直接调用gin.New创建引擎时,无需注册Recovery中间件;第二种是在调用gin.Default()创建引擎时,内部自动注册Recovery中间件。在没有使用Recovery中间件的情况下,向服务发送异常请求会导致服务端和客户端出现异常;而使用Recovery中间件后,异常被捕获并以友好的方式显示异常堆栈,同时客户端收到HTTP 错误。
Recovery中间件内部实现通过多种变体接口实现,包括CustomRecoveryWithWriter、RecoveryWithWriter、CustomRecovery以及Recovery。其中,CustomRecoveryWithWriter提供最底层的形式,允许用户自定义异常输出和恢复处理逻辑。RecoveryWithWriter则提供了Writer参数和一个可选的RecoveryFunc,如果没有定义该函数,则使用defaultHandleRecovery。CustomRecovery和Recovery则分别使用默认的DefaultErrorWriter和defaultHandleRecovery。
Recovery的核心实现通过DefaultErrorWriter和defaultHandleRecovery两个主要部分。DefaultErrorWriter负责设置日志格式为红色字体输出。defaultHandleRecovery方法是整个处理流程的核心,包含捕获、处理异常、生成响应等关键步骤。首先通过recover()方法获取panic信息,判断异常是否由客户端断开连接引起,然后获取异常堆栈、请求头,并根据异常类型和原因进行相应的处理和响应输出。最终,根据处理结果返回HTTP响应,如果是异常则返回HTTP ,如果是网络原因则使用Abort方法。
Recovery中间件的实现不仅提供了异常处理的灵活性,还确保了服务的稳定性和客户端的友好体验。通过捕获和处理异常,Recovery中间件能够有效地减少服务中断的可能性,提高系统的健壮性。总结而言,Recovery中间件在处理异常时提供了实用的方法,对于开发稳定、可靠的HTTP服务具有重要意义。