新西兰服务器

Golang中panic与recover的区别是什么


Golang中panic与recover的区别是什么

发布时间:2022-06-08 14:15:12 来源:高防服务器网 阅读:51 作者:iii 栏目:开发技术

这篇文章主要介绍“Golang中panic与recover的区别是什么”,在日常操作中,相信很多人在Golang中panic与recover的区别是什么问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Golang中panic与recover的区别是什么”的疑惑有所帮助!接下来,请跟着小编一起来学习吧!

前言

与defer类似的是,goroutine 中也有一个_panic链表头指针指向一个_panic链,发生panic的时候也是在链表头插入_panic结构体(执行gopanic)

在执行过程中发生了panic。那么panic以后的代码不会执行,转而执行panic的逻辑,再执行defer,执行到的defer要将started标记为true,同时将其defer结构体中的_panic指针指向当前的_panic,表示这个defer是由该panic触发的。再去执行defer链表,如果defer执行中还触发了panic,panic后的代码不载执行,将这个panic插入panic链头,同时将其作为当前panic。当遇到了与当前panic不符的defer,就找到该defer上的panic,将其标记为已终止,从defer链表中移除当前执行的defer。打印panic移除信息,从链表尾开始逐步输出

流程

panic执行defer的流程:

  • 先标记started=true,_panic=&panic

  • 后释放

  • 目的是为了终止之前发生的panic

异常信息的输出方式:

  • 所有还在panic链表上的项会被输出

  • 顺序与发生panic的顺序一致

// A _panic holds information about an active panic.  //  // A _panic value must only ever live on the stack.  //  // The argp and link fields are stack pointers, but don't need special  // handling during stack growth: because they are pointer-typed and  // _panic values only live on the stack, regular stack pointer  // adjustment takes care of them.  type _panic struct {      // argp 存储当前要执行的defer的函数参数地址  	argp      unsafe.Pointer // pointer to arguments of deferred call run during panic; cannot move - known to liblink  	// arg panic函数自己的参数      arg       interface{}    // argument to panic      // link,链到之前发生的panic  	link      *_panic        // link to earlier panic  	pc        uintptr        // where to return to in runtime if this panic is bypassed  	sp        unsafe.Pointer // where to return to in runtime if this panic is bypassed  	// recovered 标识panic是否被恢复      recovered bool           // whether this panic is over      // aborted 标识panic是否被终止  	aborted   bool           // the panic was aborted  	goexit    bool  }

关于recover

recover只执行一件事

  • 将当前执行的panic的recovered字段置为true

在每个defer执行完以后panic处理流程都会检查当前panic是否被recover

  • 如果当前panic已经被恢复,就会将它从panic链中移除

  • 执行到的defer也会被移除,同时要保存_defer.sp和_defer.pc

利用_defer.sp和_defer.pc跳出当前panic的处理流程,通过栈指针判断,只执行当前函数中注册的defer函数

在发生recover的函数正常结束后才会进入到检测panic是否被恢复的流程

当recover的函数又发生panic时,goroutine会将该panic加入到链头,设置为当前panic,再去执行defer链表,发现当前defer是当前panic执行的,移除当前defer,继续执行下一个,直到发现不是当前panic执行的,在panic链上找到那个panic,输出异常信息

对于已经recover标记的panic在输出异常信息时会加上recovered标记

到此,关于“Golang中panic与recover的区别是什么”的学习就结束了,希望能够解决大家的疑惑。理论与实践的搭配能更好的帮助大家学习,快去试试吧!若想继续学习更多相关知识,请继续关注高防服务器网网站,小编会继续努力为大家带来更多实用的文章!

[微信提示:高防服务器能助您降低 IT 成本,提升运维效率,使您更专注于核心业务创新。

[图文来源于网络,不代表本站立场,如有侵权,请联系高防服务器网删除]
[