Golang开发轻松学我爱编程go学习

理解go中interface关键点

2018-07-23  本文已影响12人  freelang

理解go中interface关键点


interface是golang中的精华所在,本文主要理解interface中的几个关键点。

interface即是method的集合,也是一种类型

type Inter interface {
    Get() int 
    Set(int)
}

interface变量存储的是实现类型的值


package main 

import "fmt"

type Inter interface {
    Get() int 
    Set(int)
}

type St struct {
    Age int 
}

func(s St) Get() int {
    return s.Age
} 

func(s *St) Set(age int) {
    s.Age = age 
}

func test(i Inter) {
    i.Set(10)
    fmt.Println(i.Get())
}

func main() {
    s := St{}
    test(&s)
}

这段代码中,St实现了Inter,执行test(),就完成了对Inter的使用。


s := S{}
var i Inter
i = &s
fmt.Println(i.Get())    //会自动调用S中关于Get的实现

这也体现了go中的多态性。

empty interface

func doSomething(v interface{}) {

}

既然空的interface可以接受任何类型的参数,那么一个interface{}类型的slice是不是可以接受任何类型的slice呢? --不能!

package main  

import "fmt"

func printAll(vals []interface{}) {
    for _, val := range vals {
        fmt.Println(val)
    }
}

func main() {
    names := []string{"hello", "world"}
    printAll(names)
}

执行结果:

这个例子说明go不能将slice转化成interface{}类型的slice,但是我们可以手动进行转化:

    var interfaceSlice []interface{} = make([]interface{}, len(names))
    for i,d := range names {
        interfaceSlice[i] = d
    }

执行结果:

receiver的理解

调用test(s)的执行结果如下:

这是一个错误的实现,关键在于St中Set()方法的receiver是一个pointer *St。

interface定义时并没有规定是闲着的方法receiver是value receiver 还是pointer receiver,如上述例子,当我们使用test(s)的形式调用,传递给test的是s的一份拷贝,在进行s的拷贝到Inter的转换时,s的拷贝不满速Set()方法的receiver是个pointer,也就是没有实现。

而如果反过来receiver是value,函数用pointer的形式调用:

package main 

import "fmt"

type Inter interface {
    Get() int 
    Set(int)
}

type St struct {
    Age int 
}

func(s St) Get() int {
    return s.Age
} 

func(s St) Set(age int) {
    s.Age = age 
}

func test(i Inter) {
    i.Set(10)
    fmt.Println(i.Get())
}

func main() {
    s := St{}
    test(&s)
    test(s)
}

执行结果为:

之所以没能按照我们预期的输出10 10,是因为传值不能改变原始数据的值。但是代码是能正常运行的,也就是好说receiver都value receiver,执行代码无论是pointer还是value都可以正常执行。

再思考一下出现这种现象的原因是什么呢?
如果是传入pointer,go可以根据pointer找到对应指向的值,但如果是value,传入的只能是value的拷贝temp,没办法根据value的拷贝temp去找到value原始的地址,这就是为什么pointer可以对应pointer receiver以及value receiver,但value却无法满足pointer receiver。

其实这里很关键的一点就是,实参到形参只是一个拷贝。

image
上一篇 下一篇

猜你喜欢

热点阅读