关于 interface{} 会有啥注意事项?下

简介: 我们一起来回顾一下上一次说到的 interface{}• 可以用来做多态

我们一起来回顾一下上一次说到的  interface{}

  • 可以用来做多态
  • 接口类型分为空接口类型和非空接口类型,他们的底层数据结构不太一样

这里顺便说一下,用来作态需要满足这样的条件:

  • 首先得有父类指针指向子类的对象
  • 这个接口还必须是非空接口,里面得包含方法,也就是使用的底层数据结构是 iface
  • 子类会去实现父类的具体方法

interface{} 注意断言

正常的使用断言,写一个简单的 断言 demo

type Animal interface{}
func main() {
  var a Animal = "xiaomotong"
  v, ok := a.(string)
  if !ok{
    fmt.Println("type error")
  }
  fmt.Println("v == ",v)
}

断言注意写成 2 个返回值的,一个是具体的值,一个是 bool,判断断言是否成功,若成功则说明断言正确,且 v 会被赋值为实际变量的值

切忌没头脑的强转

func main() {
  var a Animal = "xiaomotong"
  v := a.(int)
  fmt.Println("v == ",v)
}

上述这种写法,若不判断是否断言成功,直接强转,程序是会 panic 的 ,执行上述程序后效果如下:

>go run main.go
panic: interface conversion: main.Animal is string, not int
goroutine 1 [running]:
main.main()
        D:/mycode/my_new_first/interface_test/main.go:13 +0x4c
exit status 2

程序崩掉,那就是线上问题了 xdm ,这里需要注意

当然我们也可以这样写

type Animal interface{}
func main() {
  var a Animal = "xiaomotong"
  switch a.(type) {
  case string:
    fmt.Println("a type is string")
  case int:
    fmt.Println("a type is int")
  }
}

通过 switch 的方式来判断 接口的数据类型,根据不同的数据类型来做不同的事情,进行分类处理

用于反射的注意点

golang 里面有反射这个概念,简单来说,咱们可以通过反射包来实现获取接口真实的类型,和真实的数据

举个小例子:

通过反射来获取 Animal 接口的数据类型,和具体的数据值

type Animal interface{}
func main() {
  var a Animal = "xiaomotong"
  v := reflect.ValueOf(a)
  fmt.Printf("v type == ", v.Type(), " v == ", v.String())
}

执行上述代码后效果如下:

> go run main.go
v type == %!(EXTRA *reflect.rtype=string, string= v == , string=xiaomotong)

没毛病正常获取,但是如果我们这样写的话是不是也可以正确执行呢?

type Animal interface{}
func main() {
  var a Animal = "xiaomotong"
  v := reflect.ValueOf(a)
  fmt.Printf("v type == ", v.Type(), " v == ", v.Int())
}

看这个例子也就和上面的例子相差在输出的时候,一个是v.String() ,一个是v.Int() ,可是结果却相差很大

go run main.go
v type == %!(EXTRA *reflect.rtype=string, string= v == , string=xiaomotong)
D:\mycode\my_new_first\interface_test>go run main.go
panic: reflect: call of reflect.Value.Int on string Value
goroutine 1 [running]:
reflect.Value.Int(...)
        C:/Program Files/Go/src/reflect/value.go:999
main.main()
        D:/mycode/my_new_first/interface_test/main.go:26 +0x2bd
exit status 2

执行上述代码,go 程序给我们报了 panic 错误,这里需要注意 ,程序崩溃的原因是,我们将一个 string 类型的值反射成一个 Int 类型的值,直接在 反射包里面就给我们报了 panic

小知识,大挑战,简单的写 2 个注意事项,给 xdm 提个醒


欢迎点赞,关注,收藏

朋友们,你的支持和鼓励,是我坚持分享,提高质量的动力

image.png

好了,本次就到这里

技术是开放的,我们的心态,更应是开放的。拥抱变化,向阳而生,努力向前行。

我是阿兵云原生,欢迎点赞关注收藏,下次见~

相关文章
|
存储 搜索推荐 数据挖掘
使用selenium库模拟浏览器行为,获取网页的cookie值
使用selenium库模拟浏览器行为,获取网页的cookie值
|
Android开发
Android实现语音播报的两种方式
Android实现语音播报的两种方式
830 0
|
Oracle 关系型数据库 MySQL
Flink CDC产品常见问题之flink Oraclecdc 捕获19C数据时报错错如何解决
Flink CDC(Change Data Capture)是一个基于Apache Flink的实时数据变更捕获库,用于实现数据库的实时同步和变更流的处理;在本汇总中,我们组织了关于Flink CDC产品在实践中用户经常提出的问题及其解答,目的是辅助用户更好地理解和应用这一技术,优化实时数据处理流程。
|
Java 大数据 API
Java8的stream里的并行度如何使用?效率有提升吗?
Java8的stream里的并行度如何使用?效率有提升吗?
323 4
|
SQL 监控 Oracle
Oracle SQL性能优化全面指南
在数据库管理领域,Oracle SQL性能优化是确保数据库高效运行和数据查询速度的关键
1606 6
|
存储 缓存 NoSQL
缓存、分布式缓存和持久化
这篇内容介绍了缓存的概念和Redis的作用,以口袋与公文包的比喻解释了缓存如何提高数据访问速度。Redis是一个内存中的高级缓存系统,能提升系统响应速度。接着讨论了为何需要分布式缓存,通过多个“篮子”(Redis节点)解决单点故障和性能瓶颈,保证高可用性和数据安全性。最后提到了Redis的两种持久化机制——RDB(定期数据快照)和AOF(记录写操作日志),分别用照片备份和实时同步来比喻,说明它们在数据丢失风险和恢复速度上的权衡。
|
存储 负载均衡 NoSQL
【分布式技术架构】「Tomcat技术专题」 探索Tomcat集群架构原理和开发分析指南
【分布式技术架构】「Tomcat技术专题」 探索Tomcat集群架构原理和开发分析指南
423 1
|
算法 计算机视觉 Python
OpenCV高斯滤波器、双边滤波器的讲解与实战(附Python源码)
OpenCV高斯滤波器、双边滤波器的讲解与实战(附Python源码)
779 0
|
JavaScript Java 测试技术
基于ssm+vue.js+uniapp小程序的大学生创新创业平台附带文章和源代码部署视频讲解等
基于ssm+vue.js+uniapp小程序的大学生创新创业平台附带文章和源代码部署视频讲解等
118 0
|
JavaScript Shell API
笔记:Electron中关联格式、处理文件、创建链接的相关编程
本文介绍关于在Electron中绑定文件格式、在菜单中打开、使用文件API、处理桌面链接等等。
527 0

热门文章

最新文章