为了账号安全,请及时绑定邮箱和手机立即绑定

如何通过 map[string]interface{} 递归迭代

如何通过 map[string]interface{} 递归迭代

Go
慕斯709654 2023-05-08 14:30:39
我遇到了一个问题,即如何使用附加条件递归地遍历 map[string]interface{}。1) 如果一个值是一个映射 - 递归调用该方法2) 如果一个值是一个数组——调用数组的方法3)如果一个值不是地图 - 处理它。现在当方法尝试执行时doc.throughMap(mv)- 发生错误那么如何在reflect确认值是映射或数组后将某些值转换为所需的类型?type MapType map[string]interface{}type ArrayType []interface{}func (doc *Document) throughMap(docMap MapType) MapType {    for k, v := range docMap {        vt := reflect.TypeOf(v)        switch vt.Kind() {        case reflect.Map:            if mv, ok := v.(map[string]interface{}); ok {                docMap[k] = doc.throughMap(mv)            } else {                panic("error.")            }        case reflect.Array, reflect.Slice:            if mv, ok := v.([]interface{}); ok {                docMap[k] = doc.throughArray(mv)            } else {                panic("error.")            }        default:            docMap[k] = doc.processType(v)        }    }    return docMap}堆栈跟踪:panic: error. [recovered]    panic: error.goroutine 1 [running]:encoding/json.(*encodeState).marshal.func1(0xc000074cd0)    /usr/local/go/src/encoding/json/encode.go:301 +0x9apanic(0x4bd700, 0x4f9b70)    /usr/local/go/src/runtime/panic.go:513 +0x1b9project-name/package/name.(*Document).throughMap(0xc00000c028, 0xc000060180, 0xc00007e000)    /home/path/to/project/document.go:231 +0x3f4project-name/package/name.(*Document).convertDocument(0xc00000c028)    /home/path/to/project/document.go:217 +0x33project-name/pachage/name.(*Document).MarshalJSON(0xc00000c028, 0x4db740, 0xc00000c028, 0x7f3f0f7540c0, 0xc00000c028, 0xc00001c101)
查看完整描述

2 回答

?
慕妹3242003

TA贡献1824条经验 获得超6个赞

使用以下代码递归映射、数组和任何类型的切片:


func walk(v reflect.Value) {

    fmt.Printf("Visiting %v\n", v)

    // Indirect through pointers and interfaces

    for v.Kind() == reflect.Ptr || v.Kind() == reflect.Interface {

        v = v.Elem()

    }

    switch v.Kind() {

    case reflect.Array, reflect.Slice:

        for i := 0; i < v.Len(); i++ {

            walk(v.Index(i))

        }

    case reflect.Map:

        for _, k := range v.MapKeys() {

            walk(v.MapIndex(k))

        }

    default:

        // handle other types

    }

}


查看完整回答
反对 回复 2023-05-08
?
慕慕森

TA贡献1856条经验 获得超17个赞

以下是为我工作


func main() {

    x := MapType{

        "a": MapType{

            "x": MapType{

                "p": ArrayType{"l", "o", "l"},

            },

        } ,

    }

    d := &Document{}

    fmt.Println(d.throughMap(x))


}


type Document struct {}


type MapType map[string]interface{}

type ArrayType []interface{}

func (doc *Document) throughMap(docMap MapType) MapType {

    for k, v := range docMap {

        fmt.Println(k, v)

        vt := reflect.TypeOf(v)

        switch vt.Kind() {

        case reflect.Map:

            if mv, ok := v.(MapType); ok {

                docMap[k] = doc.throughMap(mv)

            } else {

                panic("error.")

            }

        case reflect.Array, reflect.Slice:

            if mv, ok := v.(ArrayType); ok {

                docMap[k] = doc.throughArray(mv)

            } else {

                panic("error.")

            }

        default:

            docMap[k] = doc.processType(v)

        }

    }

    return docMap

}


func (doc *Document) throughArray(arrayType ArrayType) ArrayType  {

    return arrayType

}


func (doc *Document) processType(x interface{}) interface{} {

    return x

}


查看完整回答
反对 回复 2023-05-08
  • 2 回答
  • 0 关注
  • 129 浏览
慕课专栏
更多

添加回答

举报

0/150
提交
取消
意见反馈 帮助中心 APP下载
官方微信