Golang断言判断值类型的实现方法

作者:鹿灏楷silves 时间:2024-02-16 09:59:38 

Golang可以通过断言,判断值的类型


s:="hello world"
i:=interface{}(s)//将数值转化为interface空接口类型
//需要注意的是,必须是空接口类型才能使用断言,如果不是空接口类型会报错
//Invalid type assertion: a.(string) (non-interface type string on left)
v,e:=i.(string)//返回value和error值,当err值为true则转化成功,value的值为括号中的值类型,当err值为false,则转化不成功

也可以通过反射判断值的类型


name:="test"
t:=relfect.TypeOf(name)
fmt.Println(t)//通过反射确定值的类型

类型断言是什么,类型断言和类型转换有什么区别,这个问题以前我也常常分不清楚。为了帮助和我有一样疑问的人,我决定得写一篇关于这方面的博文,介绍一下golang中的类型断言和类型转换的区别,在JavaScript的超集Typescript里,也同样有类型断言的概念。这篇简短的博文就是帮助大家解答这个疑问,我会尽量短的说清楚,我理解的类型转换和类型断言的区别是什么。

什么是类型转换

类型转换在很多静态类型的语言中都会有的概念,类型转换通常分为显示类型转换和隐式类型转换。强制类型转换形如:


f := 11.22
i := int(f)

例如有个float32的变量被赋值为11.22,现在我们想去掉小数部分,最简单的方法就是将float32转换为int32。

简单来说,强制类型转换就是你要从一个类型强制转换到另一个类型。适用于一些基本类型,比如int, float之类等等。但在golang中,类型匹配是相当严格的,很多时候编译器不会帮你去做,所以大多数的情况下,我们还是会做一些显示的类型转换。

比如这段看起来在其它静态类型语言中毫无问题的代码片段,在golang中编译期就会报错,golang会强制让你做类型转换。


var i int = 1
var f float64 = i

接下来看看隐式的类型转换。golang中的隐式类型转换主要存在于运行时。比如:


var w io.Writer = os.Stdout

这里将*File类型赋值给了io.Writer类型,在运行时会做一个隐式的类型转换。

什么是断言

在了解什么是类型断言之前,先来了解一下断言是什么

Golang断言判断值类型的实现方法

这是尼克杨吗?

简而言之,断言就是对一种条件进行假设,如果这是尼克杨,那么我要要干嘛?如果这不是尼克杨又怎样?
随之,类型断言就是对类型进行一种假设。

这里拿Typescript来说个事,在TS里我们很多时候会用到一种类型叫联合类型,联合类型A | B可以理解为它可以是A类型或者是B类型。实际例子:


let zhangsan:Student | null //表示zhangsan是一个Student或者null类型

我们需要使用zhangsan的时候,可以使用类型断言


if(zhangsan) zs = <Student>zhangsan
//或者
if(zhangsan) zs = zhangsan as Student

在golang中的类型断言和Typescript中的第二种类型断言比较相像。在golang中形如


x.(T)

x是一种接口类型,T可以是一种具体的类型也可以是一种接口类型

golang为什么需要类型断言

为此,我们思考一个问题,为什么Golang需要类型断言,golang中对类型的要求十分严格,而且golang中也没有Typescript中的联合类型,好像一切类型都是固定不变的,有了强制转换类型,为什么还需要类型断言呢?

在Golang中,接口类型是能够隐式转换的。看一个具体的例子:


var w io.Writer = os.Stdout

w的类型为io.Writer,但是它被赋值了*File,这是Golang帮助我们做了一次类型转换。这次类型转换是在运行时的,编译时并不能确定下来。在运行时,这个接口值的类型被赋值为了*File,与此同时,值也被赋值为了os.Stdout。

上述说明了一个问题,接口值的类型是不固定的!因为它的类型是要在运行时才能确定下来,这需要看它的动态值的类型才能确定。这就是需要类型断言的原因了。

再看一个具体的例子


var w io.Writer = os.Stdout

这条语句执行过后,w只会拥有write方法,但是原本的*File不止拥有write方法,应该还拥有read方法,同时,它也是io.ReadWriter接口的一个实例。如果这时候我们想使用read方法怎么办,那就需要类型断言了。


rw := w.(io.ReadWriter)

这里将w断言为ReadWriter类型。断言类型为一个接口。暴露了*File的read和write方法

类型断言的检查机制是怎样的

于是我们想了解Golang的类型断言的检查机制是怎样的,换句话说,Golang到底是如何来判断断言是否成功的。

首先明确的是x必须为一个接口类型,而T可以是一个具体的类型也可以是一个接口类型。下面我们分情况讨论。

1.当T为一个接口类型时

当T为一个接口时,首先会判断x的动态值是否符合T这个接口,如果符合的话,断言成功,反之断言失败,断言失败时会抛出一个panic异常。但是如果类型断言出现在一个预期有两个结果的赋值操作中,那么断言失败不会抛出panic异常,而是用一个bool值标识是否断言成功。


var w io.Writer = os.Stdout
b, ok := w.(*bytes.Buffer)

为了健壮性,我们应该对ok返回的结果进行处理。标识是否断言成功。


var w io.Writer = os.Stdout
if b, ok := w.(*bytes.Buffer);!ok {
 fmt.Fprintf(os.Stderr, "断言失败")
} else {
 //TODO
}

对一个接口类型的类型断言改变了类型的表述方式,改变了可以获取的方法集合(通常更大),但是它保护了接口值内部的动态类型和值的部分(Go Programing Language)

当T为一个具体类型时

当T为一个具体类型时,会先检查x的动态值的类型是否为T,如果为T则断言成功,如果不为T,则断言失败。

具体类型的类型断言从它的操作对象中获得具体的值(Go Programing Language)

当x为nil

最后再简单的说一下x为nil的情况,当x为nil时,那么不论断言类型是任何类型,都会断言失败

来源:https://blog.csdn.net/Xiang_lhh/article/details/113626093

标签:Golang,断言,值类型
0
投稿

猜你喜欢

  • 我的页面制作方法

    2008-03-23 13:51:00
  • Mysql安装 Navicat 出现1044/1045错误的解决方法

    2024-01-15 18:13:30
  • python 爬取华为应用市场评论

    2023-08-31 23:18:32
  • mysql中general_log日志知识点介绍

    2024-01-12 23:49:58
  • Python多线程与多进程相关知识总结

    2021-07-06 21:37:16
  • Python装饰器用法实例总结

    2022-09-27 01:26:23
  • 对Python使用mfcc的两种方式详解

    2023-06-04 14:56:52
  • mysql show processlist 显示mysql查询进程

    2024-01-19 07:42:16
  • python docx 中文字体设置的操作方法

    2021-01-12 22:33:44
  • 解析php中var_dump,var_export,print_r三个函数的区别

    2023-11-24 12:27:02
  • Python黑魔法@property装饰器的使用技巧解析

    2023-03-10 16:54:01
  • 用滤镜使网页图片产生旋转效果

    2007-11-03 11:36:00
  • python列表插入append(), extend(), insert()用法详解

    2021-05-12 13:32:40
  • Python限制内存和CPU使用量的方法(Unix系统适用)

    2023-08-02 08:41:44
  • Python中numpy数组的计算与转置详解

    2022-11-24 09:27:57
  • TensorFlow如何指定GPU训练模型

    2022-05-31 10:03:21
  • 用户反馈对产品设计的帮助

    2009-02-09 13:15:00
  • python实现播放音频和录音功能示例代码

    2023-08-20 23:23:15
  • Python调用两个机器人聊天的实战

    2021-09-30 23:10:52
  • python定义具名元组实例操作

    2023-05-16 22:22:55
  • asp之家 网络编程 m.aspxhome.com