当前位置:   article > 正文

GO 多态-Golang接口的定义/空接口/断言_go 接口 如何设置非公共部分

go 接口 如何设置非公共部分

Golang中的接口


在Go语言中接口(interface)是一种类型,一种抽象的类型。

接口(interface)定义了一个对象的行为规范, 只定义规范不实现 ,由具体的对象来实现规范的细节 。

实现接口的条件:

一个对象只要全部实现了接口中的方法 ,那么就实现了这个接口。换句话说,接口就是一个需要实现的方法列表。

为什么要使用接口


下面的代码中定义了猫和狗,然后它们都会叫,你会发现main函数中明显有重复的代码

如果我们后续再加上猪、青蛙等动物的话,我们的代码还会一直重复下去,那我们能不能把它们当成“能叫的动物”来处理呢?

  1. type Cat struct {
  2. Name string
  3. }
  4. func (c Cat) Say() string {
  5. return c.Name + ":喵喵喵"
  6. }
  7. type Dog struct {
  8. Name string
  9. }
  10. func (d Dog) Say() string {
  11. return d.Name + ": 汪汪汪"
  12. }
  13. func main() {
  14.     c := Cat{Name: "小白猫"} // 小白猫:喵喵喵
  15.     fmt.Println(c.Say())
  16.     d := Dog{"阿黄"}
  17.     fmt.Println(d.Say()) // 阿黄: 汪汪汪
  18. }
  19. /*
  20. 小白猫:喵喵喵
  21. 阿黄: 汪汪汪
  22. */

定义一个Usber接口(从下往上转换)

定义一个 Usber 接口让 Phone 和 Computer 结构体实现这个接口

  1. //1.接口是一个规范
  2. type Usber interface {
  3.     getName() string
  4. }
  5. //2.如果接口里面有方法的话,必要要通过结构体或者通过自定义类型实现这个接口
  6. type Phone struct {
  7.     Name string
  8. }
  9. type Computer struct {
  10.     Brand string
  11. }
  12. func (c *Computer) getName() string {
  13.     return c.Brand
  14. }
  15. //3.手机要实现usb接口的话必须得实现usb接口中的所有方法
  16. func (p *Phone) getName() string {
  17.     return p.Name
  18. }
  19. func transData(usber Usber) string {
  20.     name := usber.getName()
  21.      return fmt.Sprintf("%s%s", name, "处理后")
  22. }
  23. func main() {
  24.     p := &Phone{
  25.     Name: "华为手机",
  26.     }
  27.     c := &Computer{
  28.     Brand: "联想电脑",
  29.     }
  30.     var p1 Usber // golang中接口就是一个数据类型
  31.     p1 = p // 表示手机实现Usb接口
  32.     fmt.Println(p1.getName())
  33.    
  34. //接口使用场景,处理相同类型的数据
  35.     newName := transData(p)
  36.     newName1 := transData(c)
  37.     fmt.Println(newName, newName1)
  38. }

当我们要去处理同一数据类型的时候这个数据的时候,比如猫,狗是同一类数据类型。

像k8s的deployment,pod,configmap,secretd等等这些资源的时候,它都是k8s的一个资源,在获取列表的时候会对其做数据处理,比如要对其进行数据排序,那么只需要对这个接口进行排序就行了,不需要对每个资源都去写一遍排序的方法,那么这样不是很冗余吗?

空接口


1. 空接口说明

  • Golang中空接口也可以直接当做类型来使用,可以表示任意类型 (泛型概念,最大的特点

  • Golang 中的接口可以不定义任何方法,没有定义任何方法的接口就是空接口。

  • 空接口表示没有任何约束,因此任何类型变量都可以实现空接口。

  • 空接口在实际项目中用的是非常多的,用空接口可以表示任意数据类型

2. 空接口作为函数的参数

  1. //空接口作为函数的参数
  2. func show(a interface{}) {
  3. fmt.Printf("值:%v 类型:%T\n", a, a)
  4. }
  5. func main() {
  6.     show(20) // 值:20 类型:int
  7.     show("你好golang") // 值:你好golang 类型:string
  8.     slice := []int{1, 2, 34, 4}
  9.     show(slice) // 值:[1 2 34 4] 类型:[]int
  10.     }

printf println是可以接受任何参数的,func Printf(format string , a ...interface{})可以看到这里可以接受任何空接口类型,...类似于切片类型。

  1. func test(a int, b ...interface{}) {
  2. for _, v := range b {
  3. fmt.Printf("%v,%T \n", v, v)
  4. }
  5. }
  6. func main() {
  7. test(1, "a", 22, []string{"1"})
  8. }
  9. a,string
  10. 22,int
  11. [1],[]string

3. 切片实现空接口

切片和map是同一类型的元素,有了空接口,在同一个切片同一个map中就可以将任何类型的数据放进去。

  1. func main() {
  2.     var slice = []interface{}{"张三", 20, true, 32.2}
  3.     fmt.Println(slice) // [张三 20 true 32.2]
  4. }

4. map 的值实现空接口

  1. func main() { // 空接口作为 map 值
  2. var studentInfo = make(map[string]interface{})
  3. studentInfo["name"] = "张三"
  4. studentInfo["age"] = 18
  5. studentInfo["married"] = false
  6. fmt.Println(studentInfo) // [age:18 married:false name:张三]
  7. }

类型断言(从上往下转换,接口类型转化为具体类型)


  • 一个接口的值(简称接口值)是由一个具体类型和具体类型的值两部分组成的。

  • 这两部分分别称为接口的动态类型和动态值。(一个空接口,可以是结构体,指针,布尔各种类型。它其实是动态的,你传递什么都可以,这就叫做动态类型和动态值)

  • 如果我们想要判断空接口中值的类型,那么这个时候就可以使用类型断言

  • 其语法格式: x.(T)

x : 表示类型为 interface{}的变量

T : 表示断言 x 可能是的类型

断言返回两个值,一个是断言是否转化成果,一个是转化结果,注意空接口类型是空接口类型。

  1. var str interface{}
  2. str = "hello"
  3. str = str + "sss" //这样写是会报错的,因为空接口类型不是字符串类型,需要断言转化才可以
  1. type test1 struct {
  2. name string
  3. age int
  4. }
  5. func main() {
  6. var t interface{}
  7. //未断言前没有结构体的特征,需要断言转化为结构体指针
  8. t = &test1{
  9. name: "hh",
  10. age: 12,
  11. }
  12. v, ok := t.(*test1) //这里其实就是做了一个类型转化,这里的类型就变为来结构体指针类型
  13. if ok {
  14. fmt.Printf("类型:%T 值:%#v\n", v, v)
  15. } else {
  16. fmt.Println("xxx")
  17. }
  18. }
  19. 类型:*main.test1 值:&main.test1{name:"hh", age:12}

转化为具体类型之后,那么具体类型的方法是都可以去调用的,空接口是不能访问属性和方法的。转化为具体的类型之后就可以访问其属性和方法。

值接收者和指针接收者(节省内存)


1. 值接收者

当方法作用于值类型接收者时,Go语言会在代码运行时将接收者的值复制一份。

在值类型接收者的方法中可以获取接收者的成员值,但修改操作只是针对副本,无法修改接收者变量本身。

  1. type Usb interface {
  2.     Start()
  3.     Stop()
  4. }
  5. type Phone struct {
  6. Name string
  7. }
  8. func (p Phone) Start() {
  9. fmt.Println(p.Name, "开始工作")
  10. }
  11. func (p Phone) Stop() {
  12. fmt.Println("phone 停止")
  13. }
  14. func main() {
  15.     phone1 := Phone{ // 一:实例化值类型
  16.     Name: "小米手机",
  17.     }
  18.     var p1 Usb = phone1 //phone1 实现了 Usb 接口 phone1 是 Phone 类型
  19.     p1.Start()
  20.    
  21.     phone2 := &Phone{ // 二:实例化指针类型
  22.     Name: "苹果手机",
  23.     }
  24.     var p2 Usb = phone2 //phone2 实现了 Usb 接口 phone2 是 *Phone 类型
  25.     p2.Start() //苹果手机 开始工作
  26. }

2. 指针接收者

  • 指针类型的接收者由一个结构体的指针组成

  • 由于指针的特性,调用方法时修改接收者指针的任意成员变量,在方法结束后,修改都是有效的。

  • 这种方式就十分接近于其他语言中面向对象中的 this 或者 self 。

  • 例如我们为 Person 添加一个 SetAge 方法,来修改实例变量的年龄。

3. 指针类型接收者 使用时机

注:并不是所有情况下都希望修改数据

  • 需要修改接收者中的值

  • 接收者是拷贝代价比较大的大对象

  • 保证一致性,如果有某个方法使用了指针接收者,那么其他的方法也应该使用指针接收者。

从使用场景上面来说,基本上都会使用指针,如果使用了第三方的包,第三方包的某些返回值它只是一个值类型,所以第三方类型返回什么类型我们就使用什么类型,针对第三方包。

声明:本文内容由网友自发贡献,转载请注明出处:【wpsshop博客】
推荐阅读
相关标签
  

闽ICP备14008679号