Go语言学习笔记(十)
一、处理错误
- 错误处理及Go语言的独特之处
- 理解错误类型
- 创建错误
- 设置错误的格式
- 从函数返回错误
- 错误和可用性
- 慎用panic
无论在什么语言中,错误处理都是相当重要的一部分内容,有些语言在捕获到错误时引发异常。而Go语言将错误作为一种类型,这意味着可将错误传递给函数和方法。
1 错误处理及Go语言的独特之处
在Go语言中,一种约定是在调用可能出现问题地方法或函数时,返回一个类型为错误的值,这意味着如果出现问题,函数不会引发异常,而让调用者决定如何处理错误,实例如下:
package main
import (
"fmt"
"io/ioutil"
)
func main() {
file, err := ioutil.ReadFile("foo.txt")
if err != nil {
fmt.Println(err)//输出open foo.txt: The system cannot find the file specified.
return
}
fmt.Println("%s", file)
}
这个例子使用函数ioutil.Readfile
读取了文件foo.txt
中的内容,如果读取出现问题会返回一个错误值,如果没有问题,err就为nil
,这个函数的声明如下:
func ReadFile (filename string) ( []byte, error)
2 理解错误类型
在Go语言中,错误是一个值。标准库声明了接口error,如下所示。
type error interface{
Error() string
}
这个接口只有一个方法——Error,它返回一个字符串。
3 创建错误
如果要创建并返回错误,可以使用标准库中的errors包来操作
package main
import (
"errors"
"fmt"
)
func main() {
err := errors.New("Something went wrong")
if err != nil {
fmt.Println(err)
}
}
- 使用errors中的New方法创建了一个新的错误
4 设置错误的格式
除此之外,使用标准库中的fmt
包提供了方法Errorf
,可用来设置返回错误字符串的格式。这将使我们将多个值合并成更有意义的错误字符串,从而动态的创建错误字符串。
package main
import (
"fmt"
)
func main() {
name, role := "adadf", "fadfads"
err := fmt.Errorf("The %v %v quit", role, name)
if err != nil {
fmt.Println(err)
}
}
5 从函数返回错误
package main
import (
"fmt"
)
func Half(numberToHalf int) (int, error) {
if numberToHalf%2 != 0 {
return -1, fmt.Errorf("Cannot half %v", numberToHalf)
}
return numberToHalf / 2, nil
}
func main() {
n, err := Half(19)
if err != nil {
fmt.Println(err)
return
}
fmt.Println(n)
}
这个实例演示了Go语言处理错误方式的一个优点:错误处理不是在函数中,而是在调用函数的地方进行的。这在错误处理方面提供了极大的灵活性,而不是简单的一刀切。
6 错误和可用性
我们在声明错误时,要注意以下几个方面:
- 要具体地指出问题所在
- 最好提供解决问题的方案
- 对用户要有礼貌
7 慎用panic
panic
是Go语言中的一个内置函数,它终止正常的控制流程并引发恐慌(panicking),导致程序停止执行。出现普通错误时,并不提倡这种做法,因为程序将停止执行,并且没有任何余地
package main
import(
"fmt"
)
func main(){
fmt.Println("This is executed")
panic("On no. I can do no more. Goodbye.")
fmt.Println("This id not executed")
}
- 执行这个程序时,第二个输出将不会被执行,因为当执行到
panic
指令时,程序就已经强制退出了,类似于C++
的exit()
语句 - 所以不到万不得已的情况下不要使用
panic
语句 - 以下情况使用panic比较合适
- 程序处于无法恢复的状态,这可能意味着无路可走了,或者再往下执行程序将带来更多的问题
- 发生了无法处理的错误
8 相关问题
8.1 在Go代码中,if err != nil随处可见。这看起来重复太多,请问这种做法是最佳的吗?
虽然看起来重复太多,但能够检查错误实际上是Go语言的特色,这给程序员提供了极大的控制权。虽然可采用办法和第三方包来减少这样的重复,但大多数Go程序员都喜欢随心所欲的处理错误
8.2 Go支持异常吗?
不同于java等其他语言,Go语言不支持传统的try-catch-finally
控制结构,而向函数或方法的调用者报告错误。Go语言将错误作为返回值,这只是他做出的一种设计决策。
参考书籍
[1]: 【Go语言入门经典】[英] 乔治·奥尔波 著 张海燕 译