一尘不染

遍历接口

go

我想创建一个函数,该函数需要一个映射或一个数组,然后对其进行迭代,并在每个项目上调用一个函数,该函数知道如何处理遇到的任何类型。

这是我第一次失败的尝试。当前,当我在实际用例中运行它时,它总是显示“呃哦!”。

func DoTheThingToAllTheThings(data_interface interface{}) int {
    var numThings int

    switch data := data_interface.(type) {
    case map[interface{}]interface{}:
        numThings = len(data)
        for index, item := range data {
            DoTheThing(index, item)
    }
    case []interface{}:
        numThings = len(data)
        for index, item := range data {
            DoTheThing(index, item)
        }
    default:
        fmt.Println("uh oh!")
    }

    return numThings
}

数组或映射可能包含许多不同的内容,因此无法尝试匹配每个可能的输入。

换句话说, 是否有一种方法可以遍历Go中的数组或映射,而又不确切知道它是什么?


阅读 293

收藏
2020-07-02

共1个答案

一尘不染

该功能fmt.Printf("%v\n", data_interface)正是您想要的。它将打印传递给它的整个地图或数组。

您可以在此处找到实现:http :
//golang.org/src/pkg/fmt/print.go?h=printArg#L708

printArg关键点附近的线是关键:

return p.printReflectValue(reflect.ValueOf(arg), verb, plus, goSyntax, depth

它使用“ reflect”包:http :
//golang.org/pkg/reflect/来查询参数的类型。内部内容p.printReflectValuehttp
:
//golang.org/src/pkg/fmt/print.go?h=printArg#L862您将看到两种处理地图和结构的情况。然后,它使用递归printValue进行管理内容。

这是一段代码,它吸收结构的反射,然后将其转换回正确类型的另一个变量。您不能使用此方法从一种任意类型更改为另一种类型,必须在不使用反射的情况下将类型从一种强制转换为另一种是合法的。

package main

import (
    "fmt"
    "reflect"
)

type Player string

type Board struct {
    Tboard  [9]string
    Player1 Player
    Player2 Player
}

// ignore this function contents, I grabbed it from elsewhere.
func makeBoard() *Board {
    b := &Board{Tboard: [9]string{}}
    for x := 0; x < len(b.Tboard); x++ {
        b.Tboard[x] = "X"
        fmt.Println(b.Tboard[x])
    }
    b.Player1 = "George"
    b.Player2 = "Tim"

    fmt.Printf("Len: %v\n", len(b.Tboard)) // => 9

    fmt.Printf("Contents: %v\n", b)
    fmt.Printf("Syntax: %#v\n", b)
    fmt.Printf("Type: %T\n", b)
    fmt.Println("Board:", b.Tboard)
    return b
}

func main() {
    myBoard := makeBoard()

    v := reflect.ValueOf(*myBoard) // v is of type Value
    t := v.Type()

    fmt.Printf("Value: %v %T\n", v, v)
    fmt.Printf("Type:  %v %T\n", t, t)

    // would be a switch
    if t == reflect.TypeOf(*myBoard) {
        var b2 Board

        b2 = v.Interface().(Board) // cast the type interface{} into type Board
        fmt.Printf("v converted back to: %#v\n", b2)
    } else {
        fmt.Printf("t is not recognized")
    }

}

请注意,类型vmain.Board,完整的软件包名称不是Board。您要执行此操作的任何结构都必须具有导出类型,以使反射起作用。

2020-07-02