登录
首页 >  Golang >  Go问答

Golang 中的空接口

来源:stackoverflow

时间:2024-04-26 14:27:41 255浏览 收藏

目前golang学习网上已经有很多关于Golang的文章了,自己在初次阅读这些文章中,也见识到了很多学习思路;那么本文《Golang 中的空接口》,也希望能帮助到大家,如果阅读完后真的对你学习Golang有帮助,欢迎动动手指,评论留言并分享~

问题内容

编辑:这不是在 go 中使用接口的正确方法。这个问题的目的是让我了解空接口在 go 中是如何工作的。

如果go中的所有类型都实现了interface{}(空接口),为什么我无法访问catdog结构中的name字段?如何通过函数 sayhi() 访问每个结构体的 name 字段?

package main

import (
    "fmt"
)

func sayHi(i interface{}) {

    fmt.Println(i, "says hello")

    // Not understanding this error message
    fmt.Println(i.name) //  i.name undefined (type interface {} is interface with no methods)
} 

type Dog struct{
    name string
}
type Cat struct{
    name string
}

func main() {
    d := Dog{"Sparky"}
    c := Cat{"Garfield"}

    sayHi(d) // {Sparky} says hello
    sayHi(c) // {Garfield} says hello
}

解决方案


interface{} 是方法集,而不是字段集。如果类型的方法包含该接口的方法,则该类型实现该接口。由于空接口没有任何方法,因此所有类型都实现它。

如果需要访问某个字段,则必须获取原始类型:

name, ok:=i.(dog).name

如果 idog,这将恢复名称。

或者,为 dogcat 实现 getname() 函数,然后它们都将实现以下接口:

type namedtype interface {
   getname() string
}

然后你可以将你的函数重写为:

func sayhi(i namedtype) {
   fmt.println(i.getname()) 
}

您不能这样做,因为接口值不会这样做。

接口值的作用是什么——无论接口类型本身如何;接口类型是否为空并不重要,重要的是它们包含两件事:

  • 某个值的具体类型(或无类型);和
  • 该具体类型的值(或无值)。

因此,如果某个变量 v 或表达式 e 的类型为 i,其中 i 是接口类型,那么您可以使用某种语法检查这两个“字段”中的一个或两个。它们不是 struct 字段,因此您不能只使用 v.type,但您可以执行以下操作:

switch v.(type) {
case int: // the value in v has type int
case *float64: // the value in v has type float64
// etc
}

switch 中的 .(type) 表示让我看看类型字段

获取实际比较困难,因为 go 或多或少要求您首先检查类型。在您的情况下,您知道 i 持有 dogcat,因此您可以编写:

var name string
switch i.(type) {
case dog: name = i.(dog).name
case cat: name = i.(cat).name
default: panic("whatever 'i' is, it is not a dog or cat")
}
fmt.println(name)

这相当笨拙,有很多方法可以让它变得不那么笨拙,但这始终是第一步:弄清楚类型是什么

嗯,有时第一步之前还有一个步骤:弄清楚变量中是否有任何内容。你可以这样做:

if i == nil {
    ...
}

但是请注意,如果 i 中有一些类型值,并且该类型可以保存 nil 指针,则 i部分可以为 nil,但 i == nil 将为 false。这是因为 i确实有一个类型

var i interface{}
var p *int
if i == nil {
    fmt.println("i is initially nil")
}
if p == nil {
    fmt.println("p is nil")
}
i = p
if i != nil {
    fmt.printf("i is now not nil, even though i.(*int) is %v\n", i.(*int))
}

(在 Go playground 上尝试此操作)。

这通常不是使用 interface 的正确方法

大多数情况下(也有例外)我们甚至不会尝试查看某些接口的类型。相反,我们定义一个接口,提供方法(我们可以调用的函数)来完成我们需要完成的事情。请参阅 Burak Serdar's answer,其中接口类型有一个 getname 方法。然后,我们不再试图找出某人给我们的有限类型中的哪一个,而是说:

name := i.getName()

对底层具体值调用 getname 方法。如果 i 持有 dog,则调用 func (dog) getname() string,您需要定义它。如果 i 持有 cat,则它调用 func (cat) getname() string。如果您决定向集合中添加名为 bird 的类型,则可以定义 func (bird) getname() string 等。

(通常,这些方法也会导出getname,而不是 getname。)

以上就是《Golang 中的空接口》的详细内容,更多关于的资料请关注golang学习网公众号!

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