迭代一个接口

时间:2014-09-13 09:17:36

标签: types go

我想创建一个函数,它接受一个map或一个任意数组,并迭代它调用每个项目上的函数,该函数知道如何处理它遇到的任何类型。

这是我第一次失败的尝试。目前,当我在我的实际用例中运行时,它总是说'呃哦!"。

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中迭代数组或地图而不确切知道它是什么?

2 个答案:

答案 0 :(得分:1)

要使您的示例正常工作,您需要构建一个interface{}(或地图)数组,以便检测正确的类型:

// This won't work, as the .(type) would be []S
someS := []S{S{}}
DoTheThingToAllTheThings(someS)

// This will: it will go in case []interface{}:
someSI := make([]interface{}, len(someS))
for i, s := range someS {
    someSI[i] = s
}
DoTheThingToAllTheThings(someSI)

查看full example here

但这意味着您仍然可以在interface{}功能中使用DoTheThing 这里没有真正的通用,正如我在“What would generics in Go be?”中提到的那样。

答案 1 :(得分: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

使用"反映" package:http://golang.org/pkg/reflect/来查询参数的类型。在p.printReflectValue里面:http://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")
    }

}

请注意,v的类型为main.Board,即完整的包名称,而不是Board。 要执行此操作的任何结构都必须具有导出的反射类型才能工作。