登录
首页 >  Golang >  Go教程

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

来源:脚本之家

时间:2022-12-29 17:09:28 328浏览 收藏

来到golang学习网的大家,相信都是编程学习爱好者,希望在这里学习Golang相关编程知识。下面本篇文章就来带大家聊聊《Golang断言判断值类型的实现方法》,介绍一下断言、值类型,希望对大家的知识积累有所帮助,助力实战开发!

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类型,在运行时会做一个隐式的类型转换。

什么是断言

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

这是尼克杨吗?

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

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

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

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

if(zhangsan) zs = 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时,那么不论断言类型是任何类型,都会断言失败

今天关于《Golang断言判断值类型的实现方法》的内容就介绍到这里了,是不是学起来一目了然!想要了解更多关于golang的内容请关注golang学习网公众号!

声明:本文转载于:脚本之家 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>
评论列表