美文网首页
区块链 GO interface

区块链 GO interface

作者: ze__lin | 来源:发表于2018-01-25 01:45 被阅读0次
interface 是一种类型,从它的定义可以看出来用了 type 关键字,更准确的说 interface 是一种具有一组方法的类型,这些方法定义了 interface 的行为。

简单的说,interface是一组method签名的组合,我们通过interface来定义对象的一组行为。golang是通过 interface 类型接口实现的继承多态的效果。

go 允许不带任何方法的 interface ,这种类型的 interface 叫 empty interface。

如果一个类型实现了一个 interface 中所有方法,我们说类型实现了该 interface,所以所有类型都实现了 empty interface,因为任何一种类型至少实现了 0 个方法。go 没有显式的关键字用来实现 interface,只需要实现 interface 包含的方法即可。

废话不多说。上代码,请注意下面的两种写法:

package main

import "fmt"

// 这是一个接口
type Animal interface {
    bark()//声明一个方法
}
//结构体
type Dog struct {
    name string
}

func (dog *Dog) bark() {
    fmt.Printf("名字为%s的小狗在叫\n", dog.name)
}

type Pig struct {
    name string
}

func (pig *Pig) bark() {
    fmt.Printf("名字为%s的小猪在叫\n", pig.name)
}

func bark11(animal Animal) {
    animal.bark()
}

func main() {
    dog := Dog{"旺财"}
    pig := Pig{"猪八戒"}
    bark11(&dog)
    bark11(&pig)

}

输出为:
名字为旺财的小狗在叫
名字为猪八戒的小猪在叫

package main

import "fmt"

// 这是一个接口
type Animal interface {
    bark()
}

type Dog struct {
    name string
}

func (dog Dog) bark() {
    fmt.Printf("名字为%s的小狗在叫\n", dog.name)
}

type Pig struct {
    name string
}

func (pig Pig) bark() {
    fmt.Printf("名字为%s的小猪在叫\n", pig.name)
}

func bark11(animal Animal) {
    animal.bark()
}

func main() {
    dog := Dog{"旺财"}
    pig := Pig{"猪八戒"}
    bark11(dog)
    bark11(pig)

}

输出为:
名字为旺财的小狗在叫
名字为猪八戒的小猪在叫

这两种写法关系到指针的的用法,后面会详细讲解

下面再通过一个简单的案例,讲解interface的使用过程

//1
type I interface {    
    Get() int
    Set(int)
}
//2
type S struct {
    Age int
}
func(s S) Get()int {
    return s.Age
}
func(s *S) Set(age int) {
    s.Age = age
}
//3
func f(i I){
    i.Set(10)
    fmt.Println(i.Get())
}
func main() {
    s := S{} 
    f(&s)  //4
}

这段代码在 #1 定义了 interface I,在 #2 用 struct S 实现了 I 定义的两个方法,接着在 #3 定义了一个函数 f 参数类型是 I,S 实现了 I 的两个方法就说 S 是 I 的实现者,执行 f(&s) 就完了一次 interface 类型的使用。

interface 的重要用途就体现在函数 f 的参数中,如果有多种类型实现了某个 interface,这些类型的值都可以直接使用 interface 的变量存储。


s := S{}
var i I //声明 i 
i = &s //赋值 s 到 i
fmt.Println(i.Get())

不难看出 interface 的变量中存储的是实现了 interface 的类型的对象值,这种能力是 duck typing。在使用 interface 时不需要显式在 struct 上声明要实现哪个 interface ,只需要实现对应 interface 中的方法即可,go 会自动进行 interface 的检查,并在运行时执行从其他类型到 interface 的自动转换,即使实现了多个 interface,go 也会在使用对应 interface 时实现自动转换,这就是 interface 的魔力所在。

empty interface

interface{} 是一个空的 interface 类型,根据前文的定义:一个类型如果实现了一个 interface 的所有方法就说该类型实现了这个 interface,空的 interface 没有方法,所以可以认为所有的类型都实现了 interface{}。

package main

import "fmt"

type Animal interface{}

type Dog struct {
    age int
}

type Cat struct {
    weigth float64
}

func main() {
    dog := Dog{18}
    cat := Cat{300}
    dog1 := Dog{17}
    cat1 := Cat{301}

    animals := []Animal{dog, cat, dog1, cat1}
    fmt.Println(animals)
}

输出为:
[{18} {300} {17} {301}]

如果定义一个函数参数是 interface{} 类型,这个函数应该可以接受任何类型作为它的参数。
package main

import "fmt"

type Sq struct {
    l float64
}

func (sq Sq) area() float64 {
    return sq.l * sq.l
}

type Circle struct {
    r float64
}

func (c Circle) area() float64 {
    return 3.14 * c.r * c.r
}

// 接收任何的参数,但是意义不是特别大
func info(v interface{}) {
    fmt.Println(v)
}

func main() {
    sq := Sq{80.0}
    sq1 := Sq{80.0}
    sq2 := Sq{80.0}
    c := Circle{35.5}
    info(sq)
    animals := []interface{}{sq, sq1, sq2, c}
    fmt.Println(animals)
}
输出为:
{80}
[{80} {80} {80} {35.5}]

最后附一题(解题思路很好)
写一个父类把三个类中相同的属性替换掉。
创建4个dog,3个Cat,2个pig对象,将他们乱序装到一个数组里面
写一个方法,animals.sort(1) //1 或者 0,如果是1,按姓名排序,如果是0,按年龄排序

package main

import (
    "fmt"
    "strings"
)

// 接口是用来写共同的方法的
type sortInterFace interface {
    sortname() string
    sortage() int
}

type ids struct {
    name string
    age  int
}

// 猫
type cat struct {
    ids
}

func (cat cat) sortname() string {
    return cat.name
}
func (cat cat) sortage() int {
    return cat.age
}

//狗
type dog struct {
    ids
}

func (dog dog) sortname() string {
    return dog.name
}
func (dog dog) sortage() int {
    return dog.age
}

// 猪

type pig struct {
    ids
}

func (pig pig) sortname() string {
    return pig.name
}
func (pig pig) sortage() int {
    return pig.age
}

//主函数
func main() {

    testids()

}

func testids() {
    dog1 := dog{ids{"aehe1", 10}}
    dog2 := dog{ids{"hehe12", 131}}
    dog3 := dog{ids{"behe13", 132}}
    dog4 := dog{ids{"jehe14", 133}}
    //cat
    cat1 := cat{ids{"cat12", 11}}
    cat2 := cat{ids{"cat13", 12}}
    cat3 := cat{ids{"cat14", 13}}
    //pig

    pig1 := pig{ids{"pig1", 102}}
    pig2 := pig{ids{"pig2", 103}}
    // 装入到一个数组当中

    ids2 := []sortInterFace{dog1, dog2, dog3, dog4, cat1, cat2, cat3, pig1, pig2}

    fmt.Println(ids2)

    // 1,按年龄排序.. 0,按姓名排序

    // ids2 = sort(ids2... ,0)
    ids2 = sort(ids2, 0)
    // 打印结果
    for i := 0; i < len(ids2); i++ {

        fmt.Println("==== 排序后%v", ids2[i])

    }
}

func sort(nums []sortInterFace, num int) []sortInterFace {

    switch {
    case num == 1:

        return sortByAge(nums)
    case num == 0:

        return sortByName(nums)
    default:
        return nums
    }
}

// 如果等于1按age排序
func sortByAge(idss []sortInterFace) []sortInterFace {

    for i := 0; i < len(idss)-1; i++ {

        for j := 0; j < len(idss)-1-i; j++ {
            //后面的最小
            if idss[j].sortage() > idss[j+1].sortage() {

                idss[j], idss[j+1] = idss[j+1], idss[j]
            }

        }

    }
    return idss
}

// 如果等于0按姓名排序
func sortByName(idss []sortInterFace) []sortInterFace {

    for i := 0; i < len(idss)-1; i++ {

        for j := 0; j < len(idss)-1-i; j++ {
            //后面的最小 Compare(a, b string) int
            // 前面的大
            if strings.Compare(idss[j].sortname(), idss[j+1].sortname()) > 0 {

                idss[j], idss[j+1] = idss[j+1], idss[j]
            }

        }

    }
    return idss

}

相关文章

网友评论

      本文标题:区块链 GO interface

      本文链接:https://www.haomeiwen.com/subject/yzapaxtx.html