Golang之interface

Creak_Phone · · 2775 次点击 · · 开始浏览    
这是一个创建于 的文章,其中的信息可能已经有所发展或是发生改变。

一、什么是interface
        简单地说,interface是一组method的组合,可以通过interface来定义对象的一组行为。
        
interface类型
        interface类型定义了一组方法,如果某个对象实现了某个接口的所有方法,则此对象就实现了此接口。详细语法参考如下例子:
        type Human struct {
            name string
            age int
            phone string
        }
        type Student struct {
            Human        //匿名字段Human
            school string
            loan float32
        }
        type Employee struct {
            Human        //匿名字段Human
            company string
            money float32
        }
        //Human对象实现Sayhi方法
        func (h *Human) SayHi() {
            fmt.Printf("Hi, I am %s you can call me on %s\n", h.name, h.phone)
        }
        //Human对象实现Sing方法
        func (h *Human) Sing(lyrics string) {
            fmt.Println("La la, la la la, la la la la...", lyrics)
        }
        //Human对象实现Guzzle方法
        func (h *Human) Guzzle(beerStein string) {
            fmt.Println("Guzzle Guzzle Guzzle...", beerStein)
        }
        //Employee重载Human的Sayhi方法
        func (e *Employee) SayHi() {
            fmt.Printf("Hi, I am %s, I work at %s. Call me on %s\n", e.name, e.company, e.phone)
        }
        //Student实现BorrowMoney方法
        func (s *Student) BorrowMoney(amount float32) {
            s.loan += amount
        }
        //Employee实现SpendSalary方法
        func (e *Employee) SpendSalary(amount float32) {
            e.money -= amount
        }
        //定义interface
        type Men interface {
            SayHi()
            Sing(lyrics string)
            Guzzle(beerStein string)
        }
        type YoungChap interface {
            SayHi()
            Sing(song string)
            BorrowMoney(amount float32)
        }
        type ElderlyGent interface {
            SayHi()
            Sing(song string)
            SpendSalary(amount float32)
        }
        从上面代码可知,interface可以被任意的对象实现,比如:Men interface被Human、Student和Employee实现。同理,一个对象可以实现任意多个interface,比如:Student实现了Men和YonggChap两个interface。
        任意类型都实现了空interface(我们这样定义:interface{}),即包含0个method的interface。

三、interface值
        如果我们定义了一个interface的变量,那么这个变量可以存储实现了这个interface的任意类型的对象。比如:上例中定了一个Men interface类型的变量m,那么m里面可以存储Human、Student或者Employee值。下面举例说明如何使用:因为m能够持有这三种类型的对象,所以可以定义一个包含Men类型元素的slice,这个slice可以被赋予实现了Men接口的任意结构的对象:
        package main
        import "fmt"
        type Human struct {
            name string
            age int
            phone string
        }
        type Student struct {
            Human        //匿名字段
            school string
            loan float32
        }
        type Employee struct {
            Human        //匿名字段
            company string
            money float32
        }
        //Human实现Sayhi方法
        func (h Human) SayHi() {
            fmt.Printf("Hi, I am %s you can call me on %s\n", h.name, h.phone)
        }
        //Human实现Sing方法
        func (h Human) Sing(lyrics string) {
            fmt.Println("La la la la...", lyrics)
        }
        //Employee重载Human的SayHi方法
        func (e Employee) SayHi() {
            fmt.Printf("Hi, I am %s, I work at %s. Call me on %s\n", e.name.e.company, e.phone)
        }
        //Interface Men被Human、Student和Employee实现
        //因为这三个类型都实现了这了两个方法
        type Men interface {
            SayHi()
            Sing(lyrics string)
        } 
        func main() {
            mike := Student{Human{"Mike", 25, "222-222-xxx"}, "MIT", 0.00}
            paul := Student{Human{"Paul", 26, "111-222-xxx"}, "Harvard", 100}
            sam := Employee{Human{"Sam", 36, "444-222-xxx"}, "Golang Inc.", 1000}
            Tom := Employee{Human{"Sam", 36, "444-222-xxx"}, "Things Ltd.", 5000}
            //定义Men类型的变量i
            var i Men
            //i能存储Student
            i = mike
            fmt.Println("This is Mike, a Student:")
            i.SayHi()
            i.Sing("November rain")
            //i也能存储Employee
            i = Tom
            fmt.Println("This is Tom, an Employee:")
            i.SayHi()
            i.Sing("Born to be wild")
            //定义了slice Men
            fmt.Println("Let's use a slice of Men and see what happens")
            x := make{[]Men, 3}
            //T这三个都是不同类型的元素,但是他们实现了interface同一接口
            x[0], x[1], x[2] = paul, sam, mike
            for _, value := range x{
                value.SayHi()
            }
        }
        通过上面代码,可以看出interface就是一组抽象方法的集合,必须由其它非interface类型实现,而不能自我实现,Go语言通过interface实现了duck-typing,即“当看到一只鸟走起来像鸭子、游泳起来像鸭子、叫起来也像鸭子,那么这只鸟就可以被称为鸭子。”

四、空interface
        空interface即interface{}不包含任何的method,因此,所有类型都实现了空interface。空interface对于描述起不到任何作用,因为不包含任何的method,但是空interface在我们需要存储任意类型的数值时相当有用,因为它可以存储任意类型的数值,有点类似于C语言的void *类型。
        //定义a为空接口
        var a interface{}
        var i int = 5
        s := "hello world"
        //a可以存储任意类型的数据
        a = i
        a = s
        一个函数把interface{}作为参数,那么它可以接受任意类型的值作为参数,如果一个函数返回interface{},就可以返回任意类型的值。

五、interface函数参数
        interface的变量可以持有任意实现该interface类型的对象,这样我们可以通过定义interface参数让函数接受各种类型的参数。
        举例说明:fmt.Println可以接受任意类型的数据,打开fmt的源码能看到定义如下:
        type Stringer interface {
            String() string
        }
        也就是说,任何实现了String方法的类型都能作为参数被fmt.Println调用:
        package main
        import (
            "fmt"
            "strconv"
        )
        type Human struct {
            name string
            age int
            phone string
        }
        //通过这个方法Human实现了fmt.Stringer
        func (h Human) String() string {
            return " "+h.name+" - "+strconv.Itoa(h.age)+" years - "+h.phone+" "
        }
        func main() {
            Bob := Human{"Bob", 39, "000-7777-xxx"}
            fmt.Println("This Human is : ", Bob)
        }
        如果需要某个类型能被fmt包以特殊的格式输出,就必须实现Stringer这个接口。如果没有实现这个接口,fmt将以默认的方式输出。
        需要注意的是:实现了error接口的对象(即实现了Error() string的对象),使用fmt输出时,会调用Error()方法,因此不必再定义了String()方法了。

六、interface变量存储的类型
        通过前文我们知道interface变量能存储任意类型的数值,只要该类型实现了interface。如果想反向知道该变量里实际存储的类型时什么该当如何?目前常用的有两种方法:
1、Comma-ok断言
        在Go语言里,可以通过如下语法来判断interface里存储的数值类型:
        value, ok = element.(T)
        这里value就是变量的值,ok是一个bool类型,element是interface变量,T是断言的类型。如果element里面确实存储了T类型的值,则ok返回true,否则返回false。
        比如:value, ok := element.(int)
        这种方式的问题在于需要一长串的ifelse,断言的类型T越多则ifelse越多,因此,switch来了。
2、switch测试
        语法同C语言中的switch,具体如下:
        switch value := element.(type) {
            case int:
            ...
            case string:
            ...
            default:
            ...
        }   
        需要注意的是:element.(type)语法不能在switch外的任何逻辑里面使用,如果要在switch外面判断一个类型就使用comma-ok。

七、嵌入interface     
        Go语言里面真正吸引人的是其内置的逻辑语法,就像struct中匿名字段一样,实际上,相同的逻辑在interface中也成立。如果一个interface1作为interface2的一个嵌入字段,则interface2隐式的包含了interface1里面的method。从interface源码中可以看到如下定义:
        type Interface interface {
            sort.Interface    //嵌入字段sort.Interface
            Push(x interface{})    //a Push method to push elements into the heap
            Pop() interface{}    //a Pop elements that pops elements from the heap
        }
        sort.Interface其实就是嵌入字段,把sort.Interface的所有method隐式包含进来了,也就是下面三个方法:
        type Interface interface {
            //Len is the number of elements in the collection.
            Len() int
            //Less returns whether the element with index i should sort before the element with index j.
            Less(i, j int) bool
            //Swap swaps the elements with indexes i and j.
            Swap(i, j int)
        }
        另一个例子就是io包下面的io.ReadWriter,它包含了io包下面的Reader和Writer两个interface。
        //io.ReadWriter
        type ReadWriter interface {
            Reader
            Writer
        }
        
八、反射  
        Go语言实现了反射,所谓反射就是动态运行时的状态。我们一般用到的包是reflect包,官方的这篇文章详细地讲解了reflect包的实现原理——《Laws of reflection》。使用reflect一般分成三步:要去反射是一个类型的值(这些值都实现了空interface),首先需要把它转换成reflect对象(reflect.Type或者reflect.Value,根据不同情况调用不同函数)。这两种获取方式如下:
        t := reflect.TypeOf(i)    //得到类型的元数据,通过t我们能获取类型定义里面的所有元素
        v := reflect.ValueOf(i)    //得到实际的值,通过v我们获取存储在里面的值,还可以去改变值
        转化为reflect对象之后就可以进行一些操作了,即将reflect对象转化成相应的值:
        tag := t.Elem().Field(0).Tag        //获取定义在struct里面的标签
        name := v.Elem().Field(0).String()    //获取存储在第一个字段里面的值
        获取反射值能返回相应的类型和数值。
        var x float64 = 3.4
        v := reflect.ValueOf(x)
        fmt.Println("type:", v.Type())
        fmt.Println("kind is float64:", v.kind() == reflect.Float64)
        fmt.Println("value:", v.Float())
        最后,反射的字段必须是可读写的,类似于传值和传引用。按照下面的方式会产生错误:
        var x float64 = 3.4
        v := reflect.ValueOf(x)
        v.SetFloat(7.1)
        如果要修改相应的值,必须如下才行:
        var x float64 = 3.4
        p := reflect.ValueOf(&x)
        v := p.Elem()
        v.SetFloat(7.1)

有疑问加站长微信联系(非本文作者)

本文来自:CSDN博客

感谢作者:Creak_Phone

查看原文:Golang之interface

入群交流(和以上内容无关):加入Go大咖交流群,或添加微信:liuxiaoyan-s 备注:入群;或加QQ群:692541889

2775 次点击  
加入收藏 微博
暂无回复
添加一条新回复 (您需要 登录 后才能回复 没有账号 ?)
  • 请尽量让自己的回复能够对别人有帮助
  • 支持 Markdown 格式, **粗体**、~~删除线~~、`单行代码`
  • 支持 @ 本站用户;支持表情(输入 : 提示),见 Emoji cheat sheet
  • 图片支持拖拽、截图粘贴等方式上传