如何确定interface {}值的“真实”类型?


120

我没有找到使用interface{}类型的好资源。例如

package main

import "fmt"

func weirdFunc(i int) interface{} {
    if i == 0 {
        return "zero"
    }
    return i
}
func main() {
    var i = 5
    var w = weirdFunc(5)

    // this example works!
    if tmp, ok := w.(int); ok {
        i += tmp
    }

    fmt.Println("i =", i)
}

您知道使用Go's的很好的介绍interface{}吗?

具体问题:

  • 如何获得w的“真实”类型?
  • 有什么办法来获取类型的字符串表示形式吗?
  • 有什么方法可以使用类型的字符串表示形式来转换值?

Answers:


97

您的示例确实有用。这是一个简化的版本。

package main

import "fmt"

func weird(i int) interface{} {
    if i < 0 {
        return "negative"
    }
    return i
}

func main() {
    var i = 42
    if w, ok := weird(7).(int); ok {
        i += w
    }
    if w, ok := weird(-100).(int); ok {
        i += w
    }
    fmt.Println("i =", i)
}

Output:
i = 49

它使用类型断言


你是绝对正确的!谢谢!您对类型的类型字符串表示有任何见解吗?
cc年轻的

12
结帐reflect.TypeOf
2014年

@DmitriGoldring至少可以回答主题标题中的问题。这个答案不行。非常感谢你。
C4d19年

128

您还可以进行类型切换:

switch v := myInterface.(type) {
case int:
    // v is an int here, so e.g. v + 1 is possible.
    fmt.Printf("Integer: %v", v)
case float64:
    // v is a float64 here, so e.g. v + 1.0 is possible.
    fmt.Printf("Float64: %v", v)
case string:
    // v is a string here, so e.g. v + " Yeah!" is possible.
    fmt.Printf("String: %v", v)
default:
    // And here I'm feeling dumb. ;)
    fmt.Printf("I don't know, ask stackoverflow.")
}

谢谢你 但还不完全是。在示例中,如何将var w强制转换为int?
cc年轻的

3
Mue的示例执行相同的操作,但是使用类型切换而不是if语句。在'case int'中,'v'将为整数。在“情况下float64”,“V”将是一个float64等
JIMT

对。忘记了语法变量(type),它既狡猾又很酷
cc年轻,

51

您可以使用反射(reflect.TypeOf())来获取某物的类型,并且它给出的值(Type)具有String可以打印的字符串表示形式(方法)。


10
而且,如果您只想获取字符串或类型(例如,在Mue的答案中的类型切换链接的默认块中进行打印,则可以仅使用fmt“%T”格式而不是直接使用reflect。)
Dave C

16

这是一个同时使用开关和反射对通用映射进行解码的示例,因此,如果您不匹配类型,请使用反射将其找出来,然后在下一次添加类型。

var data map[string]interface {}

...

for k, v := range data {
    fmt.Printf("pair:%s\t%s\n", k, v)   

    switch t := v.(type) {
    case int:
        fmt.Printf("Integer: %v\n", t)
    case float64:
        fmt.Printf("Float64: %v\n", t)
    case string:
        fmt.Printf("String: %v\n", t)
    case bool:
        fmt.Printf("Bool: %v\n", t)
    case []interface {}:
        for i,n := range t {
            fmt.Printf("Item: %v= %v\n", i, n)
        }
    default:
        var r = reflect.TypeOf(t)
        fmt.Printf("Other:%v\n", r)             
    }
}

6

类型开关也可以与反射材质一起使用:

var str = "hello!"
var obj = reflect.ValueOf(&str)

switch obj.Elem().Interface().(type) {
case string:
    log.Println("obj contains a pointer to a string")
default:
    log.Println("obj contains something else")
}

2

我将提供一种基于将反射类型的参数传递给本地类型接收器的方式返回布尔值的方法(因为我找不到这样的东西)。

首先,我们声明类型为reflect.Value的匿名类型:

type AnonymousType reflect.Value

然后,为本地类型AnonymousType添加一个生成器,该生成器可以采用任何可能的类型(作为接口):

func ToAnonymousType(obj interface{}) AnonymousType {
    return AnonymousType(reflect.ValueOf(obj))
}

然后为我们的AnonymousType结构添加一个函数,该函数针对reflect.Kind进行断言:

func (a AnonymousType) IsA(typeToAssert reflect.Kind) bool {
    return typeToAssert == reflect.Value(a).Kind()
}

这使我们可以调用以下内容:

var f float64 = 3.4

anon := ToAnonymousType(f)

if anon.IsA(reflect.String) {
    fmt.Println("Its A String!")
} else if anon.IsA(reflect.Float32) {
    fmt.Println("Its A Float32!")
} else if anon.IsA(reflect.Float64) {
    fmt.Println("Its A Float64!")
} else {
    fmt.Println("Failed")
}

可以在这里看到更长的工作版本:https : //play.golang.org/p/EIAp0z62B7


1

有多种获取类型的字符串表示形式的方法。开关还可以与以下用户类型一起使用:

var user interface{}
user = User{name: "Eugene"}

// .(type) can only be used inside a switch
switch v := user.(type) {
case int:
    // Built-in types are possible (int, float64, string, etc.)
    fmt.Printf("Integer: %v", v)
case User:
    // User defined types work as well  
    fmt.Printf("It's a user: %s\n", user.(User).name)
}

// You can use reflection to get *reflect.rtype
userType := reflect.TypeOf(user)
fmt.Printf("%+v\n", userType)

// You can also use %T to get a string value
fmt.Printf("%T", user)

// You can even get it into a string
userTypeAsString := fmt.Sprintf("%T", user)

if userTypeAsString == "main.User" {
    fmt.Printf("\nIt's definitely a user")
}

链接到游乐场:https : //play.golang.org/p/VDeNDUd9uK6

By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.