如何使用Go漂亮地打印JSON?


191

有人知道在Go中漂亮地打印JSON输出的简单方法吗?

库存的http://golang.org/pkg/encoding/json/软件包似乎不包含此功能(编辑:确实如此,请参见接受的答案),而且快速的Google并没有发现任何明显的问题。

我正在寻找的用途既可以漂亮地打印出结果,json.Marshal又可以从任何地方格式化一个充满JSON的字符串,因此出于调试目的而更易于阅读。


警告:在我的实验中,在JSON字典中,字符串索引必须用括号括起来。因此,尽管大多数Javascript解释器都使用它{name: "value"}这样做并不好。 {"name": "value"}适用于Go JSON库功能。
彼得-恢复莫妮卡

2
@peterh我认为您将JavaScript文字语法与JSON混淆了。JSON规范(json.org)清楚地表明只允许使用字符串文字(意味着它需要用引号引起来),而JS语言对象语法则没有此限制。Go库遵循规范。
布拉德·皮博迪

Answers:


295

通过漂亮的印刷,我认为您的意思是缩进,像这样

{
    "data": 1234
}

而不是

{"data":1234}

最简单的方法是使用MarshalIndent,它可以让您指定通过indent参数缩进的方式。因此,json.MarshalIndent(data, "", " ")将使用四个空格缩进进行漂亮的打印。


17
是的,看起来就像是东西-它已经内置,只剩下在pkg doc中包含关键字“ pretty-print”,以便下一个搜索的人找到它。(将给文档维护者留下反馈意见。)Tks!
布拉德·皮博迪

38
json.MarshalIndent(data, "", "\t")如果您想要标签。
凯尔·布​​兰特

78
json.MarshalIndent(data, "", "🐱")如果你想要猫。抱歉
briiC

44
json.MarshalIndent(data, "", "\t🐱")如果您想要...虎斑猫... 对不起
达沃斯

78

如果您有一个要转换为JSON的对象,那么可以接受的答案很好。该问题还提到仅打印任何JSON字符串即可完成漂亮的打印,这就是我正在尝试做的事情。我只是想从POST请求(特别是CSP违规报告)中记录一些JSON 。

要使用MarshalIndent,您必须将Unmarshal其变成一个对象。如果您需要它,那就去做,但是我没有。如果您只需要漂亮地打印字节数组,那么plain Indent是您的朋友。

我最终得到的是:

import (
    "bytes"
    "encoding/json"
    "log"
    "net/http"
)

func HandleCSPViolationRequest(w http.ResponseWriter, req *http.Request) {
    body := App.MustReadBody(req, w)
    if body == nil {
        return
    }

    var prettyJSON bytes.Buffer
    error := json.Indent(&prettyJSON, body, "", "\t")
    if error != nil {
        log.Println("JSON parse error: ", error)
        App.BadRequest(w)
        return
    }

    log.Println("CSP Violation:", string(prettyJSON.Bytes()))
}

48

为了更好地使用内存,我想这样更好:

var out io.Writer
enc := json.NewEncoder(out)
enc.SetIndent("", "    ")
if err := enc.Encode(data); err != nil {
    panic(err)
}

难道SetIndentGET最近添加的?大多数人基本上是未知的。
chappjc

1
@chappjc SetIndent(原名Indent),显然加入2016年3月和围棋1.7,这是大约3年后,这个问题最初发布问道: github.com/golang/go/commit/... github.com/golang/go/commit/ …
aoeu

20

我对在Go中缺少将JSON编组为彩色字符串的快速,高质量的方法感到沮丧,所以我编写了自己的名为ColorJSON的Marshaller 。

有了它,您可以使用很少的代码轻松地产生这样的输出:

ColorJSON示例输出

package main

import (
    "fmt"
    "encoding/json"

    "github.com/TylerBrock/colorjson"
)

func main() {
    str := `{
      "str": "foo",
      "num": 100,
      "bool": false,
      "null": null,
      "array": ["foo", "bar", "baz"],
      "obj": { "a": 1, "b": 2 }
    }`

    var obj map[string]interface{}
    json.Unmarshal([]byte(str), &obj)

    // Make a custom formatter with indent set
    f := colorjson.NewFormatter()
    f.Indent = 4

    // Marshall the Colorized JSON
    s, _ := f.Marshal(obj)
    fmt.Println(string(s))
}

我正在为此写文档,但是很高兴分享我的解决方案。


17

编辑回想起来,这是非惯用的Go。像这样的小型辅助函数增加了额外的复杂性。通常,Go哲学倾向于将3条简单的行包含在1条棘手的行中。


正如@robyoder所提到的,这json.Indent是要走的路。以为我会添加这个小prettyprint功能:

package main

import (
    "bytes"
    "encoding/json"
    "fmt"
)

//dont do this, see above edit
func prettyprint(b []byte) ([]byte, error) {
    var out bytes.Buffer
    err := json.Indent(&out, b, "", "  ")
    return out.Bytes(), err
}

func main() {
    b := []byte(`{"hello": "123"}`)
    b, _ = prettyprint(b)
    fmt.Printf("%s", b)
}

https://go-sandbox.com/#/R4LWpkkHINhttp://play.golang.org/p/R4LWpkkHIN


7

这是我用的。如果无法漂亮地打印JSON,则仅返回原始字符串。对于打印包含JSON的HTTP响应很有用。

import (
    "encoding/json"
    "bytes"
)

func jsonPrettyPrint(in string) string {
    var out bytes.Buffer
    err := json.Indent(&out, []byte(in), "", "\t")
    if err != nil {
        return in
    }
    return out.String()
}

6

这是我的解决方案

import (
    "bytes"
    "encoding/json"
)

const (
    empty = ""
    tab   = "\t"
)

func PrettyJson(data interface{}) (string, error) {
    buffer := new(bytes.Buffer)
    encoder := json.NewEncoder(buffer)
    encoder.SetIndent(empty, tab)

    err := encoder.Encode(data)
    if err != nil {
       return empty, err
    }
    return buffer.String(), nil
}

2

Go中一款简单易用的漂亮打印机。可以通过以下方式将其编译为二进制文件:

go build -o jsonformat jsonformat.go

它从标准输入读取,写入标准输出并允许设置缩进:

package main

import (
    "bytes"
    "encoding/json"
    "flag"
    "fmt"
    "io/ioutil"
    "os"
)

func main() {
    indent := flag.String("indent", "  ", "indentation string/character for formatter")
    flag.Parse()
    src, err := ioutil.ReadAll(os.Stdin)
    if err != nil {
        fmt.Fprintf(os.Stderr, "problem reading: %s", err)
        os.Exit(1)
    }

    dst := &bytes.Buffer{}
    if err := json.Indent(dst, src, "", *indent); err != nil {
        fmt.Fprintf(os.Stderr, "problem formatting: %s", err)
        os.Exit(1)
    }
    if _, err = dst.WriteTo(os.Stdout); err != nil {
        fmt.Fprintf(os.Stderr, "problem writing: %s", err)
        os.Exit(1)
    }
}

它允许运行bash命令,例如:

cat myfile | jsonformat | grep "key"

2
package cube

import (
    "encoding/json"
    "fmt"
    "github.com/magiconair/properties/assert"
    "k8s.io/api/rbac/v1beta1"
    v1 "k8s.io/apimachinery/pkg/apis/meta/v1"
    "testing"
)

func TestRole(t *testing.T)  {
    clusterRoleBind := &v1beta1.ClusterRoleBinding{
        ObjectMeta: v1.ObjectMeta{
            Name: "serviceaccounts-cluster-admin",
        },
        RoleRef: v1beta1.RoleRef{
            APIGroup: "rbac.authorization.k8s.io",
            Kind:     "ClusterRole",
            Name:     "cluster-admin",
        },
        Subjects: []v1beta1.Subject{{
            Kind:     "Group",
            APIGroup: "rbac.authorization.k8s.io",
            Name:     "system:serviceaccounts",
        },
        },
    }
    b, err := json.MarshalIndent(clusterRoleBind, "", "  ")
    assert.Equal(t, nil, err)
    fmt.Println(string(b))
}

看起来如何


1

我有点新手,但这是我到目前为止收集的内容:

package srf

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

func WriteDataToFileAsJSON(data interface{}, filedir string) (int, error) {
    //write data as buffer to json encoder
    buffer := new(bytes.Buffer)
    encoder := json.NewEncoder(buffer)
    encoder.SetIndent("", "\t")

    err := encoder.Encode(data)
    if err != nil {
        return 0, err
    }
    file, err := os.OpenFile(filedir, os.O_RDWR|os.O_CREATE, 0755)
    if err != nil {
        return 0, err
    }
    n, err := file.Write(buffer.Bytes())
    if err != nil {
        return 0, err
    }
    return n, nil
}

这是功能的执行,只是标准

b, _ := json.MarshalIndent(SomeType, "", "\t")

码:

package main

import (
    "encoding/json"
    "fmt"
    "io/ioutil"
    "log"

    minerals "./minerals"
    srf "./srf"
)

func main() {

    //array of Test struct
    var SomeType [10]minerals.Test

    //Create 10 units of some random data to write
    for a := 0; a < 10; a++ {
        SomeType[a] = minerals.Test{
            Name:   "Rand",
            Id:     123,
            A:      "desc",
            Num:    999,
            Link:   "somelink",
            People: []string{"John Doe", "Aby Daby"},
        }
    }

    //writes aditional data to existing file, or creates a new file
    n, err := srf.WriteDataToFileAsJSON(SomeType, "test2.json")
    if err != nil {
        log.Fatal(err)
    }
    fmt.Println("srf printed ", n, " bytes to ", "test2.json")

    //overrides previous file
    b, _ := json.MarshalIndent(SomeType, "", "\t")
    ioutil.WriteFile("test.json", b, 0644)

}

0
//You can do it with json.MarshalIndent(data, "", "  ")

package main

import(
  "fmt"
  "encoding/json" //Import package
)

//Create struct
type Users struct {
    ID   int
    NAME string
}

//Asign struct
var user []Users
func main() {
 //Append data to variable user
 user = append(user, Users{1, "Saturn Rings"})
 //Use json package the blank spaces are for the indent
 data, _ := json.MarshalIndent(user, "", "  ")
 //Print json formatted
 fmt.Println(string(data))
}
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.