使用golang从标准输入读取json对象

6
如何从标准输入读取json对象? 我希望将一个json对象复制并粘贴到标准输入中,然后读取并解析它。 以下是代码:
    var input string
    _, err := fmt.Scan(&input)
    if err != nil {
        fmt.Println(err)
        continue
    }

    var record MedicalRecord
    if err := json.Unmarshal([]byte(input), &record); err != nil {
        log.Println(err)
        continue
    }

错误信息将被打印到控制台。

> 2018/06/26 00:26:32 invalid character ':' after top-level value
> 2018/06/26 00:26:32 unexpected end of JSON input
> 2018/06/26 00:26:32 invalid character ':' after top-level value
> 2018/06/26 00:26:32 invalid character ',' after top-level value
> 2018/06/26 00:26:32 invalid character ':' after top-level value
> 2018/06/26 00:26:32 invalid character ',' after top-level value
> 2018/06/26 00:26:32 invalid character ':' after top-level value
> 2018/06/26 00:26:32 invalid character ',' after top-level value
> 2018/06/26 00:26:32 invalid character ':' after top-level value
> 2018/06/26 00:26:32 invalid character ',' after top-level value

如果我没错的话,Go会一直读取直到找到'\n'。那么我该如何解决这个问题呢?
2个回答

12

使用 *json.Decoder 从 io.Reader 消费 JSON:

package main

import (
    "encoding/json"
    "log"
    "os"
)

type MedicalRecord struct{}

func main() {
    var record MedicalRecord

    err := json.NewDecoder(os.Stdin).Decode(&record)
    if err != nil {
        log.Fatal(err)
    }
}

您可以通过重复调用Decode来消耗多个连续的JSON文档:

package main

import (
    "encoding/json"
    "io"
    "log"
    "os"
)

type MedicalRecord struct{}

func main() {
    var record MedicalRecord

    dec := json.NewDecoder(os.Stdin)
    for {
        err := dec.Decode(&record)
        if err == io.EOF {
            return
        }
        if err != nil {
            log.Fatal(err)
        }
    }
}

2

当您调用 json.Unmarshal 时,应该拥有完整的 JSON 对象,否则您只是试图解析一些没有意义的部分对象。

  • 从 stdin 读取到足以容纳整个 JSON 对象的缓冲区中。
  • 反序列化 JSON 对象。

例如:

package main

import (
    "bytes"
    "io"
    "fmt"
    "os"
    "bufio"
    "encoding/json"
)


func main() {

    var buf bytes.Buffer
    reader := bufio.NewReader(os.Stdin)

    for {
        line, err := reader.ReadString('\n')
        if err != nil {
            if err == io.EOF {
                buf.WriteString(line)
                break // end of the input
            } else {
                fmt.Println(err.Error())
                os.Exit(1) // something bad happened
            }   
        }   
        buf.WriteString(line)

    }   

    fmt.Printf("valid json? %v\n", json.Valid(buf.Bytes()))

    type MedicalRecord struct {
        Name string `json:"name"`
        Age int `json:"age"`
    }   

    var record MedicalRecord
    err := json.Unmarshal(buf.Bytes(), &record)
    if err != nil {
        fmt.Println(err.Error())
        os.Exit(1) // something bad happened
    }   

    fmt.Printf("name: %s, age: %d\n", record.Name, record.Age)
}

如果将bufio.NewReader(file)替换为bufio.NewReader(os.Stdin),则goroutine会被阻塞。 - GianMS
我整理了代码以展示一个完整的工作程序。如果你需要启动单独的 goroutine,你将不得不自己完成。 - Matt

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接