我想创建一个函数,该函数需要一个映射或一个数组,然后对其进行迭代,并在每个项目上调用一个函数,该函数知道如何处理遇到的任何类型。
这是我第一次失败的尝试。当前,当我在实际用例中运行它时,它总是显示“呃哦!”。
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中的数组或映射,而又不确切知道它是什么?
该功能fmt.Printf("%v\n", data_interface)正是您想要的。它将打印传递给它的整个地图或数组。
fmt.Printf("%v\n", data_interface)
您可以在此处找到实现:http : //golang.org/src/pkg/fmt/print.go?h=printArg#L708
printArg关键点附近的线是关键:
printArg
return p.printReflectValue(reflect.ValueOf(arg), verb, plus, goSyntax, depth
它使用“ reflect”包:http : //golang.org/pkg/reflect/来查询参数的类型。内部内容p.printReflectValue:http : //golang.org/src/pkg/fmt/print.go?h=printArg#L862您将看到两种处理地图和结构的情况。然后,它使用递归printValue进行管理内容。
p.printReflectValue
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。您要执行此操作的任何结构都必须具有导出类型,以使反射起作用。
v
main.Board
Board