1.nil
nil相当于java里面的null,前面说过,任何类型在没有赋值的时候都会有一个零值。那么
nil就是指针、切片、map和接口的零值。
对一个值为nil的变量实施解引用的话会导致程序崩溃。
func main() {
var test *string
fmt.Println(test) //nil
fmt.Println(*test)//panic: runtime error: invalid memory address or nil pointer dereference
}
所以这个时候对test变量进行一个判空即可保证安全。
2.错误处理
在一般编程语言中,一个函数往往只返回一个值。所以当我们执行一段程序以后,返回的值往往不够可靠。因为我们不知道是否发生了错误,只能将返回值设置成一个特殊值或者空值来判断,这肯定是不适合的。在go语言里,因为可以返回多个值处理错误,所以这让一切都变得简单。
func main() {
err := creatFile("test.txt")
if err != nil {
fmt.Println(err)
os.Exit(1)
}
}
func creatFile(name string) error {
f, err := os.Create(name)
if err != nil {
return err
}
_, err = fmt.Fprintln(f, "Hello world")
f.Close()
return err
}
上面简单的做了一个判断错误,并返回正确的返回值。
3.defer
对于上面刚刚的代码中,大家有没有思考一个问题,如果写入的文字过多,那么每次是否都要去在return前面写一个f.close()呢,这样不仅重复劳动,也容易造成犯错。
所以这个时候需要使用defer了。
defer 在程序里面是使用先进后出的顺序。对于一些需要关闭的资源,因为有defer,所以就再也不会忘记去关闭释放了。因为可以直接写在开启的代码后面。当程序要结束运行时,代码里的defer定义的语句就会按队尾顺序依次出栈。
func main() {
fmt.Println(1)
fmt.Println(2)
defer fmt.Println(3)
defer fmt.Println(4)
defer fmt.Println(5)
defer fmt.Println(6)
fmt.Println(7)
//打印结果
//1 2 7 6 5 4 3
}
根据上面的代码就很好的理解了defer对语句执行顺序的影响了。
4.panic和recover
对于异常的处理,go语言提供了panic来跳出程序,显示出崩溃异常。
go语言的错误值机制促使开发者考虑错误,而不是像处理异常那样默认将其忽略。
但是go语言也提供了处理panic引发崩溃的函数recover。recover只有在defer定义的方法或函数中才能使用。recover的作用相当于java里的catch。
func main() {
defer func() {
if err := recover(); err != nil {
fmt.Println(err)
}
}()
panic("这是一个崩溃,但是程序还是正常运行了")
}
最后程序还是会正常运行并打印了panic中的崩溃提示。