golang nil、error 与interface 总结

interface{} 与nil

golang 的interface{}大体可以描述为两个元素(type+data),只有当type和data都为nil的时候
interface才会与nil相等。

var v *T
var i interface{}
i = v

v 为空指针,所以必定等于nil
interface{},type和data都是nil,所以也为nil
但是 i=v后,i的type不为nil了,所以此时不为nil

    var a interface{}
    fmt.Println(a == nil) //true
    a = "1" //data="1" type=string
    fmt.Println(a == nil) //false
    var b *int
    a = b //data=nil type=int
    fmt.Println(a == nil) //false
    fmt.Println(b == nil) //true
    fmt.Println(b == a) //true

为什么a、b与nil之间的等价不具有传递性。
1. b==nil是正确的,在a=b的时候做了个封箱的操作,在a==b的时候做了拆箱的操作
2. golang在type和interface之间的转换不明显,容易让人误解。

error

error 其实就是一种特殊的interface{}

type error interface {
    Error() string
}

example:

package main

import (
    "fmt"
)

type MyError struct {
    errCode uint8
}
func (e *MyError) Error() string {
    switch e.errCode {
    case 1:
        return "file not found"
    case 2:
        return "time out"
    case 3:
        return "permission denied"
    default:
        return "unknown error"
    }
}

// 相当于使用interface进行封箱操作
func GetError1() error {
    var err *MyError = nil
    return err
}

// 没有使用interface进行封箱操作
func GetError2() *MyError {
    var err *MyError = nil
    return err
}

func main() {
    ret1 := GetError1() //封箱,所以不为nil
    fmt.Println(ret1 == nil) //false
    e, _ := ret1.(*MyError) //显式拆箱,所以为nil
    fmt.Println(e == nil) //true

    ret2 := GetError2() // 不用转换,所以为nil
    fmt.Println(ret2 == nil) //true
}

猜你喜欢

转载自blog.csdn.net/qq_17612199/article/details/80042338