GO语言基础4 Errors 报错

更多个人笔记见:
github个人笔记仓库
gitee 个人笔记仓库

个人学习,学习过程中还会不断补充~ (后续会更新在github上)

文章目录

自己创建错误

error 信息可以自己定义

GO 复制代码
func main() {
	arg := 2
    arg2 := 4
	err := makeTea(arg)
	if err != nil {
		fmt.Println(err)
	}
	err2 := makeTea(arg2)
	if err2!= nil {
		fmt.Println(err2)
	}
}

//两种创建错误的方式
func makeTea(arg int) error {
	if arg == 2 {
		return fmt.Errorf("no more tea available")
	} else if arg == 4 {
		return errors.New("no more power available")
	}
	return nil
}

panic和recover

基础示范

panic 会报错然后程序返回,recover 结合defer 放在最后匿名函数中执行用于接受 panic

Go 复制代码
package main

import (
	"fmt"
)

func mayPanic() {
	panic("a problem") //手动 panic
}
func main() {
	defer func() {
		if r := recover(); r != nil {
			fmt.Println("成功接收Recovered. Error:\n", r)
		}
	}()
	mayPanic()
	fmt.Println("After mayPanic()") // won't output
}
//output:
//成功接收Recovered. Error:
//a problem
panic的返回

通过嵌套函数理解返回方式

Go 复制代码
func a(){
	defer fmt.Println("defer a")
	b()
	fmt.Println("after a")
}
func b(){
	defer fmt.Println("defer b")
	c()
	fmt.Println("after b")
}
func c(){
	defer fmt.Println("defer c")
	panic("this is a panic")
	fmt.Println("after c")
}

func main() {
	a()
}
// defer c
// defer b
// defer a
// panic: this is a panic

修改b部分:可以保证回到a

GO 复制代码
func b() {
	defer func() { //变为立即执行的函数
		fmt.Println("defer b")
		if x := recover(); x != nil {
			fmt.Println("run time panic: %v \n", x)
		}
	}()
	c()
	fmt.Println("after b")
}
//defer c
// defer b
// run time panic: %v 
//  this is a panic
// after a
// defer a

能回到 A 就会先打印出 after a 然后再结束就打印出 defer a

用于保证g()函数执行的中间件
GO 复制代码
func protect(g func()) {
	defer func(){
		log.Println("done")
		if x:=recover();x!=nil{
			log.Printf("run time panic:%v",x)
		}
	}()
	log.Println("start")
	g()   //执行g   如果有问题会返回上去的
}
panic和recover的嵌套顺序
打印最先 panic

如果在defer调用的函数中也有panic,会打印最早的panic (没有recover的时候

GO 复制代码
func main() {
	a()
}

func a() {
	defer b()
	panic("a panic")
}

func b() {
	defer fb()
	panic("b panic")
}

func fb() {
	panic("fb panic")
}
// panic: a panic
// panic: b panic
// panic: fb panic

这个例子中,先 panic a 然后调用 b,继续向下迭代

recover 捕获最近 panic
GO 复制代码
func main() {
	defer catch("main")  //增加捕获
	a()
}

func catch(name string) {
	if r := recover(); r != nil {
		fmt.Println(name, "recover:", r)
		//println会打印地址
	}
}

func a() {
	defer b()
	panic("a panic")
}

func b() {
	defer fb()
	panic("b panic")
}

func fb() {
	defer catch("fb")    // //增加捕获
	panic("fb panic")
}

// fb recover: fb panic
// main recover: b panic

执行顺序: a 中 panic ->defer 调用 b ->b 中 panic->调用 fb->fb

中 panic->defer 调用 catch,抓住的是 fb panic ->回到 main 中 defer->抓住的是b panic (因为 fb 的 panic 已经被捕获,会向前推)

相关推荐
十月不到底4 分钟前
在 Tomcat 中通过 web.xml 配置缓存控制响应头
后端
HoroMin4 分钟前
在Spring Boot中自定义JSON返回日期格式的指南
java·开发语言·spring boot·注解
涡能增压发动积5 分钟前
实践出真知,大模型也会犯糊涂
后端
汪洪墩10 分钟前
使用Mars3d加载热力图的时候,出现阴影碎片
开发语言·前端·javascript·vue.js·cesium
木木黄木木10 分钟前
使用Three.js创建炫酷的3D玻璃质感动态效果
开发语言·javascript·3d
@小红花30 分钟前
Python从入门到精通
开发语言·python
androidwork37 分钟前
Kotlin实现文件上传进度监听:RequestBody封装详解
android·开发语言·kotlin
Ghostbaby41 分钟前
eBPF 基础知识-如何构建BTF
后端
SimonKing1 小时前
集合的处理:JDK和Guava孰强孰弱?
java·后端·算法
Java微观世界1 小时前
Java逻辑运算符完全指南:短路与、非短路或、异或的妙用,一篇搞定!
后端