Golang Interface 接口
什么是 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 技术交流群。
上一篇: MySQL 中的锁介绍和使用
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论