如何在Go语言中将字节/uint8数组转换为JSON数组?

20

我有一个包含 []uint8 成员的结构体,我使用 json.Marshal 写入它。问题是,它将 uint8 解释为 char,并输出一个字符串而不是数字数组。

如果是 []int,我可以让它工作,但如果可以避免,我不想分配和复制项目。我能做到吗?


正如我在问题中所说,我正在使用json.Marshal。 我的问题不是关于 fmt,而是关于 json 库。 - Joe
我曾经有一个评论问我如何输出它,作者在我回答后删除了它。不妨保留我的回复。 - Joe
1个回答

30
根据文档[]byte将被编码为Base64字符串。

"数组和切片值编码为JSON数组,除了[]byte编码为base64编码的字符串,nil切片编码为null JSON对象。"

因此,我认为您可能需要使您的结构体实现Marshaler接口,通过实现自己的MarshalJSON方法,使您的[]uint8更理想地编码为JSON数组。
以这个例子为例:
import "fmt"
import "encoding/json"
import "strings"

type Test struct {
    Name  string
    Array []uint8
}

func (t *Test) MarshalJSON() ([]byte, error) {
    var array string
    if t.Array == nil {
        array = "null"
    } else {
        array = strings.Join(strings.Fields(fmt.Sprintf("%d", t.Array)), ",")
    }
    jsonResult := fmt.Sprintf(`{"Name":%q,"Array":%s}`, t.Name, array)
    return []byte(jsonResult), nil
}

func main() {
    t := &Test{"Go", []uint8{'h', 'e', 'l', 'l', 'o'}}

    m, err := json.Marshal(t)
    if err != nil {
        fmt.Println(err)
    }
    fmt.Printf("%s", m) // {"Name":"Go","Array":[104,101,108,108,111]}
}

http://play.golang.org/p/Tip59Z9gqs


或许更好的想法是创建一个新类型,以[]uint8作为其基础类型,并将该类型设置为Marshaler,然后在您的结构体中使用该类型。

import "fmt"
import "encoding/json"
import "strings"

type JSONableSlice []uint8

func (u JSONableSlice) MarshalJSON() ([]byte, error) {
    var result string
    if u == nil {
        result = "null"
    } else {
        result = strings.Join(strings.Fields(fmt.Sprintf("%d", u)), ",")
    }
    return []byte(result), nil
}

type Test struct {
    Name  string
    Array JSONableSlice
}

func main() {
    t := &Test{"Go", []uint8{'h', 'e', 'l', 'l', 'o'}}

    m, err := json.Marshal(t)
    if err != nil {
        fmt.Println(err)
    }
    fmt.Printf("%s", m) // {"Name":"Go","Array":[104,101,108,108,111]}
}

http://play.golang.org/p/6aURXw8P5d


1
太棒了。我认为第二个选项更具有未来性。非常感谢。 - Joe
@Joe:不客气。我更新了一下,使其更完整,通过将nil切片处理为null - the system
但是程序没有正确地打印字母? - Mayur Agarwal

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