Декодировать очень большой JSON в массив структур - PullRequest
0 голосов
/ 22 января 2019

У меня есть веб-приложение, которое имеет REST API, получает JSON в качестве входных данных и выполняет преобразования этого JSON.

Вот мой код:

func (a *API) getAssignments(w http.ResponseWriter, r *http.Request) {

   var document DataPacket
   err := json.NewDecoder(r.Body).Decode(&document)
   if err != nil {
       a.handleJSONParseError(err, w)
      return
   }

   // transformations

JSON, который я получаю - это коллекция структур. Внешнее приложение использует мое приложение и отправляет мне очень большие файлы JSON (300-400 МБ). Декодирование этого json в один момент времени занимает очень много времени и объема памяти.

Есть ли способ поработать с этим json в качестве потоковых и декодирующих структур из этой коллекции один за другим?

1 Ответ

0 голосов
/ 22 января 2019

Сначала прочитайте документацию.


Пакет JSON

import "кодировка / json"

func (* Decoder) Decode

func (dec *Decoder) Decode(v interface{}) error

Decode считывает следующее значение в формате JSON из своего ввода и сохраняет его в значении, указанном v.

Пример (Stream): В этом примере используется декодер для декодирования потокового массива JSON объекты.

Детская площадка: https://play.golang.org/p/o6hD-UV85SZ

package main

import (
    "encoding/json"
    "fmt"
    "log"
    "strings"
)

func main() {
    const jsonStream = `
    [
        {"Name": "Ed", "Text": "Knock knock."},
        {"Name": "Sam", "Text": "Who's there?"},
        {"Name": "Ed", "Text": "Go fmt."},
        {"Name": "Sam", "Text": "Go fmt who?"},
        {"Name": "Ed", "Text": "Go fmt yourself!"}
    ]
`
    type Message struct {
        Name, Text string
    }
    dec := json.NewDecoder(strings.NewReader(jsonStream))

    // read open bracket
    t, err := dec.Token()
    if err != nil {
        log.Fatal(err)
    }
    fmt.Printf("%T: %v\n", t, t)

    // while the array contains values
    for dec.More() {
        var m Message
        // decode an array value (Message)
        err := dec.Decode(&m)
        if err != nil {
            log.Fatal(err)
        }

        fmt.Printf("%v: %v\n", m.Name, m.Text)
    }

    // read closing bracket
    t, err = dec.Token()
    if err != nil {
        log.Fatal(err)
    }
    fmt.Printf("%T: %v\n", t, t)

}
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...