下篇:一文玩转Go接口

简介: 下篇:一文玩转Go接口
  1. 空接口既然可以存储任意类型的值,那么从空接口获取到的值是否可以直接使用?看下面栗子
package main
import (
 "fmt"
)
var a interface{}
var b interface{}
func main() {
 a = 1024
 b = 100
 res := a + b
 fmt.Println(res)
}

报错:

invalid operation: operator + not defined on a (variable of type interface{}) (exit status 2)

程序报错的原因:因为空接口的类型是不可以直接使用的,需要警告类型断言转换方可使用。

  1. 这次我们使用类型断言来将接口类型转成int类型
package main
import (
 "fmt"
)
var a interface{}
var b interface{}
func main() {
 a = 1024
 b = 100
 val1, res1 := a.(int)
 fmt.Println(val1, res1)
 val2, res2 := b.(int)
 fmt.Println(val2, res2)
 //val1和val2接收转换后的值,res1和res2是类型断言的状态(成功或失败),断言成功是true,反之false
}

输出:

1024 true
100 true
  1. 类型断言新姿势:当使用一个值接受断言结果时,则会直接返回断言后的值
package main
import (
 "fmt"
)
var a interface{}
var b interface{}
func main() {
 a = 1024
 b = 100
    //类型断言转换
 a1 := a.(int)
 b1 := b.(int)
    //转换后进行相加,就不会报错了
 res := a1 + b1
 fmt.Println(res)
}
  1. 体会一下使用类型断言转换失败的快感
package main
import (
 "fmt"
 "log"
)
var a interface{}
func main() {
 a = 1024
 if a1, r := a.(string); r {
  fmt.Println(a1)
 } else {
  log.Fatalln("类型断言转换失败")
 }
}

输出:

2022/10/25 10:30:48 类型断言转换失败

变量a存储值是整型,视图使用类型断言将其转换为字符串,结果报错了,这么玩是不行的,玩不起。

  1. 类型断言+switch实现数据类型判断
package main
import (
 "fmt"
)
func TestFunc(value interface{}) {
 switch value.(type) {
 case int:
  fmt.Printf("value=%v Type Int\n", value)
 case float32:
  fmt.Printf("value=%v Type Float32\n", value)
 case float64:
  fmt.Printf("value=%v Type Float64\n", value)
 case string:
  fmt.Printf("value=%v Type string\n", value)
 }
}
func main() {
 TestFunc("hello")
 TestFunc(100)
 TestFunc(89.12)
}

输出:

value=hello Type string
value=100 Type Int
value=89.12 Type Float64
  1. 还可以将接口类型转换成另一个接口类型,下面的栗子是将A接口转换成B接口
package main
import (
 "fmt"
)
type A interface{}
type B interface{}
func main() {
 var a A = 100
 b := a.(B)
 fmt.Println(b)
}

在之前的栗子,都是将接口类型转换成基本的数据类型,而这个栗子是将一个自定义的接口类型转换成另一个自定义的接口类型。

  1. 还可以将接口类型转成指针类型,看下面的栗子
package main
import "fmt"
func main() {
    //定义接口类型的变量ainter
 var ainter interface{}
 num := 100
 ainter = &num //将地址赋给接口变量
 v, r := ainter.(*int)
 fmt.Println(v, r)
}

上面的栗子中,使用类型断言将接口类型转成了int指针类型

  1. 接口可以嵌套吗?实战告诉你
package main
import "fmt"
type action1 interface {
 insert()
}
type action2 interface {
 delete()
}
type actionInterface interface {
 action1
 action2
 query()
}
type Db struct {
 Data string
}
func (d Db) insert() {
 fmt.Print("插入数据...", d.Data)
}
func (d Db) delete() {
 fmt.Print("删除数据...", d.Data)
}
func (d Db) query() {
 fmt.Print("查询数据...", d.Data)
}
func main() {
 d := Db{Data: "hello"}
 d.query()
 d.delete()
 d.insert()
}

通过上面的实战,接口是可以嵌套的,注意了,只有实现接口中所有的方法(含所有嵌套接口里的所有方法),那么才算是真正实现了接口。

  1. 实现error接口中的Error方法,来玩一个自定义错误类型的栗子
package main
import (
 "fmt"
)
type AddError struct {
 ErrorMsg string
}
func (m AddError) Error() string {
 return fmt.Sprintf("Add error %v", m.ErrorMsg)
}
func add(a int, b int) (int, error) {
 if a == 0 || b == 0 {
  errinfo := fmt.Sprintf("a=%v, b=%v", a, b)
  return 0, AddError{ErrorMsg: errinfo}
 } else {
  return a + b, nil
 }
}
func main() {
 res, err := add(8, 0)
 fmt.Println(res, err)
}

上面的栗子中,已经隐式的实现了error接口中的Error方法

  1. 如果不玩自定义的错误类型,也可以直接使用errors.New方法返回一个错误信息
package main
import (
 "errors"
 "fmt"
)
func add(a int, b int) (int, error) {
 if a == 0 || b == 0 {
  return 0, errors.New("不能为0")
 } else {
  return a + b, nil
 }
}
func main() {
 res, err := add(9, 1)
 fmt.Println(res, err)
}
相关文章
|
6月前
|
自然语言处理 Go 索引
Go语言学习8-接口类型
【4月更文挑战第1天】本篇 Huazie 向大家介绍 Go 语言中的接口类型
40 2
Go语言学习8-接口类型
|
6月前
|
设计模式 存储 监控
《Go 简易速速上手小册》第4章:接口与抽象(2024 最新版)(上)
《Go 简易速速上手小册》第4章:接口与抽象(2024 最新版)
67 1
|
6月前
|
Go
|
5月前
|
Go 数据安全/隐私保护
go 基于gin编写encode、decode、base64加密接口
go 基于gin编写encode、decode、base64加密接口
38 2
|
2月前
|
存储 Go
Go to Learn Go之接口
Go to Learn Go之接口
28 7
|
6月前
|
程序员 Go
|
3月前
|
存储 缓存 NoSQL
在 Go 中使用接口进行灵活缓存
在 Go 中使用接口进行灵活缓存
|
3月前
|
XML 存储 JSON
在Go中使用接口:实用性与脆弱性的平衡
在Go中使用接口:实用性与脆弱性的平衡
|
3月前
|
SQL 安全 测试技术
[go 面试] 接口测试的方法与技巧
[go 面试] 接口测试的方法与技巧
|
3月前
|
存储 安全 程序员