我很好奇为什么Go 不会隐式转换[]T
为[]interface{}
,而Go
会隐式转换T
为interface{}
。我缺少这种转换的重要内容吗?
例:
func foo([]interface{}) { /* do something */ }
func main() {
var a []string = []string{"hello", "world"}
foo(a)
}
go build
抱怨
不能在函数参数中使用(类型[]字符串)作为类型[] interface {}
如果我尝试明确执行此操作,则会遇到同样的事情:b := []interface{}(a)
抱怨
无法将(类型[]字符串)转换为类型[] interface {}
因此,每次我需要进行这种转换时(这似乎很多),我一直在做这样的事情:
b = make([]interface{}, len(a), len(a))
for i := range a {
b[i] = a[i]
}
是否有更好的方法来执行此操作,或者有标准的库函数来帮助进行这些转换?每次我想调用一个可以接收例如int或字符串列表的函数时,多写4行代码似乎有点愚蠢。
在Go中,有一个通用规则,即语法不应隐藏复杂/昂贵的操作。将a转换string
为an
interface{}
需要O(1)时间。由于切片仍然是一个值[]string
,interface{}
因此也需要O(1)时间将a转换为an
。但是,将a转换[]string
为an的时间[]interface{}
为O(n),因为切片的每个元素都必须转换为an interface{}
。
此规则的一个例外是转换字符串。当将a string
与a
[]byte
或a进行相互[]rune
转换时,即使转换是“语法”,Go仍然可以执行O(n)。
没有标准的库函数可以为您完成此转换。您可以使用反射创建一个,但是它比三行选项要慢。
反射示例:
func InterfaceSlice(slice interface{}) []interface{} {
s := reflect.ValueOf(slice)
if s.Kind() != reflect.Slice {
panic("InterfaceSlice() given a non-slice type")
}
ret := make([]interface{}, s.Len())
for i:=0; i<s.Len(); i++ {
ret[i] = s.Index(i).Interface()
}
return ret
}
最好的选择是使用问题中给出的代码行:
b := make([]interface{}, len(a))
for i := range a {
b[i] = a[i]
}
我有一个界面,看起来像 公共接口密钥检索器{ } 我希望实现是这样的 CustomerTypeKeyRetriever(实现类) 我该如何实现这一点。目前它抛出一个编译错误——“类型CustomerTypeKeyRetriever必须实现继承的抽象方法KeyRetriever.getKey(Object)”
问题内容: 我对接口和类实现接口有疑问。 这是我的代码: 我想创建一个这样的实例: 可以,但是如何从接口实例中调用?这是安全的吗? 感谢帮助。 问题答案: 是的,就可以了(如果你改变的声明来实现),并作为参考真的很安全,只要 不 参考的实例。 但是,这通常是个坏主意。使用接口的全部目的是能够与 任何 实现一起使用- 它是将抽象与实现分开。如果你然后将需要一个具体的实现,你还不如让的类型正好与开始。
问题内容: 在进行中,是否可以通过某种方式动态地转换变量? 例如,如果简单的转换将是: 如果我事先不知道年龄是整数怎么办?一种简单的书写方式是 有没有办法实现这样的目标?反射包提供了一些在运行时确定或强制转换类型的方法-但我找不到像上述提到的东西(适用于所有类型的通用方案)。 问题答案: 不,你不能。Go是一种静态类型的语言。变量的类型在编译时确定。 如果要动态确定的,可以使用类型切换:
主要内容:类型断言的格式,将接口转换为其他接口,将接口转换为其他类型Go语言中使用接口断言(type assertions)将接口转换成另外一个接口,也可以将接口转换为另外的类型。接口的转换在开发中非常常见,使用也非常频繁。 类型断言的格式 类型断言是一个使用在接口值上的操作。语法上它看起来像 i.(T) 被称为断言类型,这里 i 表示一个接口的类型和 T 表示一个类型。一个类型断言检查它操作对象的动态类型是否和断言的类型匹配。 类型断言的基本格式如下: 其中,i
7.2. 接口类型 接口类型具体描述了一系列方法的集合,一个实现了这些方法的具体类型是这个接口类型的实例。 io.Writer类型是用的最广泛的接口之一,因为它提供了所有的类型写入bytes的抽象,包括文件类型,内存缓冲区,网络链接,HTTP客户端,压缩工具,哈希等等。io包中定义了很多其它有用的接口类型。Reader可以代表任意可以读取bytes的类型,Closer可以是任意可以关闭的值,例如一
问题内容: 我正在尝试在Java中使用lambda和流,但我对此很陌生。当我尝试创建lambda表达式时,我在IntelliJ中收到了此错误“ lambda转换的目标类型必须是接口” 难道我做错了什么? 问题答案: 我怀疑这只是Java的类型推断不够聪明。尝试