3 回答
TA贡献1831条经验 获得超10个赞
对于类型断言(您使用的),只有实际类型很重要。所以somethingFuncy
只等于somethingFuncy
和不等于func(int) bool
。
解释
首先,这与铸造无关。在 go 中没有铸造。有类型断言和类型转换。
您正在处理类型断言并假设与类型转换相同的条件。我在阅读您的问题时犯了同样的错误,但实际上行为存在巨大差异。
假设您有两种类型,例如int
和type MyInt int
。它们是可转换的,因为它们共享相同的基础类型(转换规则之一),所以这是有效的(play):
var a int = 10 var b MyInt = MyInt(a)
现在,假设a
不是类型int
而是类型interface{}
(play):
var a interface{} = int(10) var b MyInt = MyInt(a)
编译器会告诉你:
无法将 (type interface {}) 转换为 MyInt 类型:需要类型断言
所以现在我们不再做转换而是断言。我们需要这样做(播放):
var a interface{} = int(10) var b MyInt = a.(MyInt)
现在我们遇到了与您的问题相同的问题。此断言因恐慌而失败:
恐慌:接口转换:接口是int,而不是main.MyInt
其原因在规范的类型断言部分中有说明:
对于接口类型和类型的表达式 x
T
,主表达式x.(T)
断言x
不是,nil
并且存储在其中的值x
是类型T
。该符号x.(T)
称为类型断言。 更准确地说,如果T
不是接口类型,x.(T)
则断言 的动态类型x
与类型相同T
。
所以int
必须与 相同MyInt
。类型标识的规则规定(除其他规则外):
如果两个命名类型的类型名称源自相同的 TypeSpec,则它们是相同的。
由于int
并MyInt
具有不同的声明(TypeSpecs),它们不相等并且断言失败。当您断言a
to 时int
,断言有效。所以你在做什么是不可能的。
奖金:
实际检查发生在这段代码中,它只是检查两种类型是否如预期的一样。
TA贡献1851条经验 获得超4个赞
使用 Go 1.9 中的类型断言,您可以简单地添加=
定义类型的位置。
type somethingFuncy = func(int) bool
这告诉编译器这somethingFuncy
是func(int) bool
.
TA贡献1802条经验 获得超5个赞
只是为了完成 nemo 很棒的答案,请注意,虽然您不能直接从interface{}
给定动态类型(例如,int
)的接口(ef, )跳转到另一种类型(例如,type MyInt int
),但您可以一个接一个地执行两个步骤:
断言变量的动态类型是您期望的;
将该断言的结果转换为您选择的类型。
请注意,由于底层类型,顾名思义,是dynamic,因此最好测试类型断言是成功还是失败。另一方面,类型转换的正确性是由编译器强制执行的。
这是您的游乐场片段略有修改:http : //play.golang.org/p/FZv06Zf7xi
- 3 回答
- 0 关注
- 286 浏览
添加回答
举报