Golang Interface 接口

发布于 2022-02-03 12:51:20 字数 3112 浏览 1104 评论 0

什么是 interface

关于 interface(接口)这种抽象类型,只需要记住并且理解这一句最关键的话:

一个类型如果拥有一个 interface 需要的所有方法,那么这个类型就实现了这个 interface

比如我们自定义一种 interface 类型:

type SortItem interface {
    Len() int
    Less(i, j int) bool
    Swap(i, j int)
}

我们的 SortItem 包含了 Len()Less()Swap() 三个方法,那么,只要有某个类型实现了自己的 Len()Less()Swap() 方法,这个类型就可以看作是这个interface,比如我们自己定义了一种类型 ItemPrice

type ItemPrice []int32

func(p *ItemPrice) Len() int {
    return len(p)
}
func(p *ItemPrice) Less(i, j int) bool {
    return p[i] < p[j]
}
func(p *ItemPrice) Swap(i, j int) {
    p[i], p[j] = p[j], p[i]
}

我们不止可以用例子里的 []int32 类型去实现这些方法,结构体或者其他的类型也行。可以看到我们的 ItemPrice 类型实现了 Len()Less()Swap() 方法,因此,它就可以作为 SortItem 这种 interface 被使用:

var temp SortItem // 定义了一个SortItem类型的变量
var priceList ItemPrice // 定义了一个ItemPrice的变量
temp = priceList

如果有一个函数的输入参数是 SortItem 这种 interface,那么我们就可以把我们实现的 ItemPrice 类型的变量传入这个函数,比如:

func Sort(data SortItem) {
    // ...
    // do quickSort(data)...
}

其实,golang 的 sort 包就是用类似的方式实现的排序,可以对我们自定义的类型进行排序,可以发现,好处就在于,不同类型我们需要的比较方式,即 Less() 方法可能是不一样的,而通过 interface,sort 包里的 Sort 函数可以对不同的类型按照我们需要的方式进行排序。

如何描述一个 interface

在go语言内部,一个 interface 类型的变量实际上存储了两个值:该变量存储的值(value)和这个值的实际类型(type):

interface{} 类型

之前我们说过, 一个类型如果拥有一个 interface 需要的所有方法,那么这个类型就实现了这个 interface。 那么,对于 interface{} 这种类型来说,它的内部不包含任何方法,因此可以认为所有的类型天然就实现了 interface{} 这种类型。

在我们需要存储任意类型的数值的时候, interface{} 相当有用

当定义一个变量的时候,如果不能确定这个变量是什么类型,就可以将其定义为一个 interface{},比如一个函数的输入输出参数,定义为interface{}就可以接收和返回任意的类型;比如定义一个slice或者map的时候,可以让slice里面存不同的类型。举例:

var m1 map[string]interface{}
m1["name"] = "XXX"  // value可以是string类型
m1["age"] = 24  // value可以是int类型
m1["male"] = true  // value可以是bool类型

类型断言

也许我们定义了一个 interface{} 类型的变量之后可以一路用下去,但总会遇到有些时候需要将它转换成我们想要的特定类型比如 int32,这个时候可以使用类型断言(type assertion),用来判断变量是什么类型,比如下面这个例子:

func ConvertF32(i interface{}) float32 {
   var fi float32
   switch i.(type) {
   case float32:
      fi = i.(float32)
   case float64:
      fi = float32(i.(float64))
   case int32:
      fi = float32(i.(int32))
   case int64:
      fi = float32(i.(int64))
   case int:
      fi = float32(i.(int))
   case string:
      fj, _ := strconv.ParseFloat(i.(string), 64)
      fi = float32(fj)
   }
   return fi
}

另一种写法是:

var i interface{}
var j int32
val, ok := i.(int32)
if ok {
    j = val
}

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。
列表为空,暂无数据

关于作者

JSmiles

生命进入颠沛而奔忙的本质状态,并将以不断告别和相遇的陈旧方式继续下去。

文章
评论
84963 人气
更多

推荐作者

微信用户

文章 0 评论 0

小情绪

文章 0 评论 0

ゞ记忆︶ㄣ

文章 0 评论 0

笨死的猪

文章 0 评论 0

彭明超

文章 0 评论 0

    我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
    原文