Чтение данных в интерфейсе map [string] {} - PullRequest
0 голосов
/ 15 апреля 2019

Исходный сервер возвращает данные в формате Json для нескольких объектных интерфейсов. Как мы можем проанализировать такие данные?

Я использую переменную типа JSON map [string] interface {} для хранения результата изсервер

 The data return from Server.
"data": [
       {
        "group": "PAA_TEST",
        "id": "2018-04-10T09:24:18.000000Z",
        "name": "PAA_STATION",
        "released": true,
        "version": 33
    },
    {
        "group": "PAA_TEST",
        "id": "2018-03-19T10:50:21.000000Z",
        "name": "PAA_STATION",
        "released": false,
        "version": 32
    }

my fmt.print output outputdata ["data"] // где выходные данные представляют собой карту JSON [string] interface {}

    [
       map[group:PAA_TEST id:2018-04-10T09:24:18.000000Z name:PAA_STATION 
       released:true version:33] 
       map[group:PAA_TEST id:2018-03-19T10:50:21.000000Z name:PAA_STATION 
       released:false version:32] 
   ]

Как можно выполнить итерацию снесколько интерфейсов карты?Например, если я просто хочу обработать информацию с освобожденным статусом как true.Я пытаюсь использовать разные методы для индексации, но пока не повезло.

1 Ответ

0 голосов
/ 16 апреля 2019

Лучшее решение - декодировать JSON напрямую в тип Go, соответствующий структуре данных.Это позволяет избежать утверждений типа, необходимых для копания в map[string]interface{}.

. Я предполагаю, что общая функция выглядит примерно так:

func request(path string, ... more arguments) (map[string]interface{}}, error) {
      ...
      resp, err := client.Do(req)
      if err != nil {
         return nil, err
      }
      defer resp.Body.Close()
      ...
      var result map[string]interface{}
      err := json.NewDecoder(resp.Body).Decode(&result)
      return result, err
}

Измените функцию, чтобы взять указатель нарезультат в качестве аргумента:

func request(pv interface{}, path string, ... more arguments) error {
      ...
      resp, err := client.Do(req)
      if err != nil {
         return err
      }
      defer resp.Body.Close()
      ...
      err := json.NewDecoder(resp.Body).Decode(pv)
      return err
}

Вызовите эту модифицированную функцию следующим образом:

var result struct { 
    Data []struct{ 
        Group, ID, Name string
        Released bool
        Version int 
    }
}
err := request(&result, "some/path", ... more arguments )
if err != nil {
    // handle error
}

for _, d := range result.Data {
   if !d.Released {
       continue
   }
   fmt.Println(d.Group, d.ID, d.Name, d.Version)
   ... process d
}
...