嵌入式技术
1、defer语句
defer 语句的用途是:含有 defer 语句的函数或类型方法,会在该函数或方法将要返回之前(return之前),调用另一个函数。所以,****我们可以在函数开始时,将一些容易忘记的结束操作用defer声明好,这样保证这些结束操作能在函数结束后正确执行。当defer要调用的函数有参数时,执行 defer 语句的时候,就会对延迟函数的实参进行求值。举例如下
func printA(a int) {
fmt.Println("value of a in deferred function", a) } func main() { a := 5 defer printA(a) //此时a=5,所以实际传到printA中的参数值为5,输出为5 a = 10 fmt.Println("value of a before deferred function call", a) }
当一个函数内多次调用 defer 时,Go 会把 defer 调用放入到一个栈中,随后按照后进先出(Last In First Out, LIFO)的顺序执行。
2、错误处理
2.1、error类型
在 Go 中,错误一直是很常见的。错误用内建的 error 类型来表示。就像其他的内建类型(如 int、float64 等),错误值可以存储在变量里、作为函数的返回值等等。error类型定义如下:
type error interface { //是一个接口,实现该接口的类型都可以当作错误类型 Error() string }
使用示例:
func main() { f, err := os.Open("/test.txt") if err != nil {//error类型 fmt.Println(err) return } fmt.Println(f.Name(), "opened successfully")//println会自动调用Error()输出错误信息 }
2.2、提取错误信息的方法
Error()返回的描述不是很详细,我们可以用下面的方法获取更详细的错误信息:
判断底层结构体类型(即实现error接口的结构体),使用结构体字段获取更多信息
判断底层结构体类型(即实现error接口的结构体),调用类型的方法获取更多信息
与error类型变量直接比较
func main() {
files, err := filepath.Glob("[") if error != nil && err == filepath.ErrBadPattern {//比较,判断是不是某个指定的错误类型 fmt.Println(err) return } fmt.Println("matched files", files) }
2.3、fmt.Errorf
fmt 包中的 Errorf 函数会根据格式说明符,规定错误的格式,并返回一个符合该错误的字符串。示例:
func circleArea(radius float64) (float64, error) {
if radius < 0 { //返回错误类型 return 0, fmt.Errorf("Area calculation failed, radius %0.2f is less than zero", radius) } return math.Pi * radius * radius, nil } func main() { radius := -20.0 area, err := circleArea(radius) if err != nil { fmt.Println(err) return } fmt.Printf("Area of circle %0.2f", area) }
3 panic和recover
有些情况,当程序发生异常时,无法继续运行。在这种情况下,我们会使用 panic 来终止程序。当函数发生 panic 时,它会终止运行,在执行完所有的defer延迟函数后,程序控制返回到该函数的调用方。这样的过程会一直持续下去,直到当前协程的所有函数都返回退出,然后程序会打印出panic信息,接着打印出堆栈跟踪(Stack Trace),最后程序终止。当程序发生 panic 时,使用 recover 可以重新获得对该程序的控制。 需要注意:
你应该尽可能地使用错误,而不是使用 panic 和 recover。只有当程序不能继续运行的时候,才应该使用 panic 和 recover 机制。
**必须在defer函数中直接调用recover**。在延迟函数内调用 recover,可以取到 panic 的错误信息,并且停止panic续发事件(Panicking Sequence),程序运行恢复正常。如果在延迟函数的外部调用 recover,就不能停止panic续发事件。
只有在相同的Go协程中调用 recover 才管用。recover 不能恢复一个不同协程的 panic
使用示例
func recoverName() { if r := recover(); r!= nil {//恢复程序运行 fmt.Println("recovered from ", r) debug.PrintStack()//打印panic异常的堆栈信息 } } func fullName(firstName *string, lastName *string) { defer recoverName() //延迟函数 if firstName == nil { panic("runtime error: first name cannot be nil")//触发程序异常 } if lastName == nil { panic("runtime error: last name cannot be nil") } fmt.Printf("%s %s ", *firstName, *lastName) fmt.Println("returned normally from fullName") } func main() { defer fmt.Println("deferred call in main") firstName := "Elon" fullName(&firstName, nil) fmt.Println("returned normally from main") }
一个通用的recover defer使用的模板,可供参考:
func foo() (err error) { defer func() { if r := recover(); r != nil { switch x := r.(type) { case string: err = errors.New(x) case error: err = x default: err = fmt.Errorf("Unknown panic: %v", r) } } }() panic("TODO") }
编辑:黄飞
全部0条评论
快来发表一下你的评论吧 !