golang:interface

2017-12-29  本文已影响0人  程序员饭饭

how to avoid gotchas explain the internal mechinasm of interface

接口在golang中的内部实现其实是一个包含meta信息的结构体和一个数据指针。而空接口的内部实现与之有差异,包含表示类型的静态类型和一个数据指针。
// interface类型的内部实现
type iface struct {
        tab  *itab
        data unsafe.Pointer
}

type itab struct {
        inter  *interfacetype
        _type  *_type
        link   *itab
        hash   uint32 // copy of _type.hash. Used for type switches.
        bad    bool   // type does not implement interface
        inhash bool   // has this itab been added to hash?
        unused [2]byte
        fun    [1]uintptr // variable sized
}
//interface{}的内部实现
type eface struct {
        _type *_type
        data  unsafe.Pointer
}
下面的代码里说明了可能出现的gotachs。
// foo() == nil    true
func foo() error {
    var err1 error   //nil
    return err1
}

// foo() == nil     false
func foo() error {
    var err2 *os.PathError   //nil
    return err2 
}
两段代码中err的结构不一样

第一段代码:


err1

第二段代码:


err2
我们可以将任何类型的数据赋值给空接口,但是下面的代码
// 编译器将报错cannot use []int literal (type []int) as type []interface {} in return argument
func foo() []interface {
     return []int{1, 2, 3, 4, 5}
}
来看看[]interface和[]int实际是怎么存储的
[]int转成[]interface
// 重写后的代码
func foo() []interface {
    var s []interface
    for _, elem := range []int{1, 2, 3, 4, 5} {
        s = append(s, elem)
    }
    return s
}
有这样一个问题,如果一个类型T或者T,都可以使用该类型的变量调用receiver类型是T或者T的方法,也就是说无论是T还是T调用某方法时,编译器会将该类型自动转换为receiver的类型。但是在接口的赋值中,如果T实现了接口的method set,那么既可以将T类型变量赋给接口变量,也可以将T类型赋给接口变量。而如果是T实现了接口的method set,那么只能将T的变量赋给接口变量,而不能将T类型的变量赋给接口变量。我们来看例子:
// T或者*T类型的变量都可以调用 func (t *T) show()
type T string

func (t *T) show() {
    fmt.Println(t)
}

var (
    t1 *T
    t2 T
)
t1.show()  // call func (t *T) show()
t2.show()  // call func (t *T) show(),编译器自动将t2转换为&t2 
// T或者*T类型的变量都可以调用 func (t T) show()
type T string

func (t T) show() {
    fmt.Println(t)
}

var (
    t1 *T
    t2 T
)
t1.show()  // call func (t T) show(),编译器自动将t1类型转换为*t1
t2.show()  // call func (t T) show()
// 赋值给接口变量的T或者*T类型的变量都可以调用 func (t T) show()
type T string

type I interface {
    show()
}

func (t T) show() {
    fmt.Println(t)
}

var (
    t1 *T
    t2 T
    i I
)

i = t1
i.show()  // call func (t T) show()
i = t2
i.show()  // call func (t T) show()
// 赋值给接口变量的T或者*T类型的变量中,只有*T可以赋给接口变量
type T string

type I interface {
    show()
}

func (t *T) show() {
    fmt.Println(t)
}

var (
    t1 *T
    t2 T
    i I
)

i = t1
i.show()  // call func (t T) show()
i = t2
i.show()  // Error, type T not implements I

第四个栗子中使用T类型变量无法赋值给接口变量,因为T类型的变量没有实现接口中的方法。为什么呢?因为T的method set是receiver type是T method set与T method set的总和;反之则不行。golang官方文档method type中有说明。

上一篇下一篇

猜你喜欢

热点阅读