当前位置: 首页 > news >正文

广州哪个区最好seo网站页面优化

广州哪个区最好,seo网站页面优化,长沙装修公司名单,制作音乐排行榜html5使用的go版本为 go1.21.2 首先我们写一个简单的panic调度与捕获代码 package mainfunc main() {defer func() {recover()}()panic(panic test) }通过go build -gcflags -S main.go获取到对应的汇编代码 可以看到当我们调度panic时#xff0c;Go的编译器会将这段…使用的go版本为 go1.21.2 首先我们写一个简单的panic调度与捕获代码 package mainfunc main() {defer func() {recover()}()panic(panic test) }通过go build -gcflags -S main.go获取到对应的汇编代码 可以看到当我们调度panic时Go的编译器会将这段代码翻译为CALL runtime.gopanic(SB) 我们先来看一下panic构造体的底层源码 panic源码与解读 //代码位置 $GOROOT/src/runtime/runtime2.go L:1035type _panic struct {argp unsafe.Pointer // 指向在 panic 运行期间执行的延迟调用参数的指针不可移动 - liblink 工具已知其位置arg any // 参数link *_panic // panic链表pc uintptr // 返回到运行时的位置sp unsafe.Pointer // 返回到运行时的栈指针位置recovered bool // 是否已被恢复aborted bool // 是否已被中止goexit bool // 是否执行了 Goexit 函数 }gopanic源码与解读 //代码位置 $GOROOT/src/runtime/panic.go L:826 // 实现预声明函数 panic func gopanic(e any) {// 处理异常参数为 nil 的情况if e nil {// 如果 debug.panicnil 不等于 1将e设置为PanicNilError类型//if debug.panicnil.Load() ! 1 {e new(PanicNilError)} else {panicnil.IncNonDefault()}}// 获取当前的Ggp : getg()// 判断当前M上运行的G是不是当前Gif gp.m.curg ! gp {print(panic: )printany(e)print(\n)throw(panic on system stack)}// malloc过程中出现panicif gp.m.mallocing ! 0 {print(panic: )printany(e)print(\n)throw(panic during malloc)}// 禁止抢占的情况下执行 panic (! 保持当前G在这M运行)if gp.m.preemptoff ! {print(panic: )printany(e)print(\n)print(preempt off reason: )print(gp.m.preemptoff)print(\n)throw(panic during preemptoff)}// 当初M处于锁的状态if gp.m.locks ! 0 {print(panic: )printany(e)print(\n)throw(panic holding locks)}// 定义一个panic变量var p _panicp.arg e //这个e 就是我们panic(xxxx) 里面写的东西//将这个panic加入到G的_panic链表中去p.link gp._panic gp._panic (*_panic)(noescape(unsafe.Pointer(p))) // 增加运行panic延迟计数runningPanicDefers.Add(1)// 计算 getcallerpc/getcallersp以避免扫描 gopanic 帧addOneOpenDeferFrame(gp, getcallerpc(), unsafe.Pointer(getcallersp()))for {//逐步获取当前G中的defer调用d : gp._defer// 如果获取到的构造体为空直接返回。if d nil {break}// 如果当前_defer运行将_defer从G的延迟链表移除释放对应的_defer构造体资源防止重复执行if d.started {if d._panic ! nil {d._panic.aborted true}d._panic nilif !d.openDefer {d.fn nilgp._defer d.linkfreedefer(d)continue}}// 标记当前_defer为运行状态d.started true// 记录_defer的panicd._panic (*_panic)(noescape(unsafe.Pointer(p)))done : trueif d.openDefer { //如果_defer使用了 open-coded defers编码的延迟调用// 运行open-coded defer函数done runOpenDeferFrame(d) //如果当前栈下面没有其他延迟函数则返回trueif done !d._panic.recovered { //panic没有recoveraddOneOpenDeferFrame(gp, 0, nil)}} else {//执行对应方法//getargp返回其caller的保存callee参数的地址p.argp unsafe.Pointer(getargp()) d.fn()}p.argp nilif gp._defer ! d {throw(bad defer entry in panic)}d._panic nilpc : d.pcsp : unsafe.Pointer(d.sp)if done { //将_defer从G的延迟链表移除释放对应的_defer构造体资源d.fn nilgp._defer d.linkfreedefer(d)}if p.recovered { //panic已经恢复gp._panic p.link if gp._panic ! nil gp._panic.goexit gp._panic.aborted {// A normal recover would bypass/abort the Goexit. Instead,// we return to the processing loop of the Goexit.gp.sigcode0 uintptr(gp._panic.sp)gp.sigcode1 uintptr(gp._panic.pc)mcall(recovery)throw(bypassed recovery failed) // mcall should not return}runningPanicDefers.Add(-1)// 从G中获取一个_defer构造体d : gp._defervar prev *_deferif !done { //如果未执行完毕跳过当前的帧直接执行下一个prev dd d.link}for d ! nil {if d.started { //如果启动退出循环break}if d.openDefer { //如果使用了 open-coded defersif prev nil { //将_defer从G的延迟链表移除释放_defergp._defer d.link} else {prev.link d.link}newd : d.linkfreedefer(d)d newd} else {prev dd d.link}}gp._panic p.link //上面有对应的赋值又重新赋了一遍没啥用for gp._panic ! nil gp._panic.aborted { //循环G中的_panic链表去掉已经被标记中止的_panicgp._panic gp._panic.link}if gp._panic nil { // 如果当前G没有panic, 重置信号为0gp.sig 0}// 将恢复帧发送给recovery.gp.sigcode0 uintptr(sp)gp.sigcode1 pcmcall(recovery)throw(recovery failed) // mcall should not return}}// 没有更多的延迟调用现在采用传统的 panic 方式// 由于在冻结世界之后调用任意用户代码是不安全的// 我们调用 preprintpanics 来调用所有必要的 Error// 和 String 方法以在 startpanic 之前准备好 panic 字符串。preprintpanics(gp._panic)fatalpanic(gp._panic) //触发致命的 panic*(*int)(nil) 0 //为了消除编译器的错误提示 }当我们调度recover时Go的编译器会将这段代码翻译为CALL runtime.gorecover(SB) gorecover源码与解读 //代码位置 $GOROOT/src/runtime/panic.go L:1045 func gorecover(argp uintptr) any {gp : getg() //获取当前Gp : gp._panic // 从当前G中获取一个_panic// 如果G存在panic它的状态不为中止还未进行painc捕获函数调用参数相同if p ! nil !p.goexit !p.recovered argp uintptr(p.argp) {p.recovered truereturn p.arg}return nil }总结 从上面的源码我们可以了解到panic的大致逻辑当使用panic关键词时将painc加入到G的_panic链表中去. 调度时 defer func() {recover()}()会改写_painc中的recovered字段可恢复的panic必须要recover的配合。 而且这个recover必须位于同一goroutine的直接调用链上否则无法对 panic 进行恢复未写完有些细节点还是没读懂后续查阅资料补充。
http://www.lakalapos1.cn/news/60574/

相关文章:

  • 义乌做网站公司义乌网站制作公司企业邮箱网址
  • 网站与网站自动跳转代码一流的镇江网站建设
  • 杭州大的做网站的公司教育培训机构排名
  • 网站建设制作设计公司哪家好吉林3厅官齐聚任免大会宁波网站建设
  • 安徽建站优化建设银行投资网站
  • 有哪些做封面的网站邯郸网站设计哪家好
  • 广东网站建设联系电话万网主机建wordpress
  • 做网站兼容性如何处理规模以上工业企业的标准
  • 豪车网站建设背景asp网站出现乱码
  • 綦江建设银行网站网站开发技术网站
  • 网站建设有趣名称网站空间使用方法
  • 永嘉网站建设工作室c 做网站后端
  • 深圳网站设计美工医院网站建设的指导思想
  • 东莞一站式网站建设青海做网站找谁
  • 哪个网站有利于做课件成都热点新闻最新
  • 做网站什么行业前景好怎么安装的wordpress主题
  • 建设局网站作用国内工业设计网站
  • 网站制作好吗做谷歌推广的网站如何引流
  • 网站默认首页怎么做网站建设价格对比单
  • 门户网站建设工作讲话个人网站备案做淘宝客
  • 平凉网站建设redu怎么在Front做网站
  • 快速让百度收录网站html 写wordpress
  • 中山网站建设制作 .超凡科技好用建站模板
  • 响应式网站 价格承德公司网站建设
  • 网站百度商桥asp网站源码 怎么安装
  • 网站设计作业平台wordpress 附件id
  • 做网站的资源有哪些学校网站 asp
  • 网站建设com网站开发大数据库
  • 诚讯通网站wordpress自动留言
  • 天河微网站建设返利商城网站怎么做