时间:2021-05-22
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.22i := int(f)例如有个float32的变量被赋值为11.22,现在我们想去掉小数部分,最简单的方法就是将float32转换为int32。
简单来说,强制类型转换就是你要从一个类型强制转换到另一个类型。适用于一些基本类型,比如int, float之类等等。但在golang中,类型匹配是相当严格的,很多时候编译器不会帮你去做,所以大多数的情况下,我们还是会做一些显示的类型转换。
比如这段看起来在其它静态类型语言中毫无问题的代码片段,在golang中编译期就会报错,golang会强制让你做类型转换。
var i int = 1var 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 = <Student>zhangsan//或者if(zhangsan) zs = zhangsan as Student在golang中的类型断言和Typescript中的第二种类型断言比较相像。在golang中形如
x.(T)x是一种接口类型,T可以是一种具体的类型也可以是一种接口类型
为此,我们思考一个问题,为什么Golang需要类型断言,golang中对类型的要求十分严格,而且golang中也没有Typescript中的联合类型,好像一切类型都是固定不变的,有了强制转换类型,为什么还需要类型断言呢?
在Golang中,接口类型是能够隐式转换的。看一个具体的例子:
var w io.Writer = os.Stdoutw的类型为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断言判断值类型内容请搜索以前的文章或继续浏览下面的相关文章希望大家以后多多支持!
声明:本页内容来源网络,仅供用户参考;我单位不保证亦不表示资料全面及准确无误,也不保证亦不表示这些资料为最新信息,如因任何原因,本网内容或者用户因倚赖本网内容造成任何损失或损害,我单位将不会负任何法律责任。如涉及版权问题,请提交至online#300.cn邮箱联系删除。
失败的类型断言,返回的值为最近断言类型的零值代码入下:funcmain(){vardatainterface{}="ehoo"ifres,ok:=data.(i
断言单元测试框架的核心是断言方法,通常叫assert()。该方法通常接收一个值--需要断言的值,以及一个表示该断言目的的描述。如果该值执行的结果为true,断言
Pythonassert语句,又称断言语句,可以看做是功能缩小版的if语句,它用于判断某个表达式的值,如果值为真,则程序可以继续往下执行;反之,Python解释
断言的概念断言用于证明和测试程序的假设,比如“这里的值大于5”。断言可以在运行时从代码中完全删除,所以对代码的运行速度没有影响。断言的使用断言有两种方法:一种是
JavaScript本身没有判断一个变量是不是空值的函数,因为变量有可能是string,object,number,boolean等类型,类型不同,判断方法也不