我有一个包含 []uint8
成员的结构体,我使用 json.Marshal
写入它。问题是,它将 uint8
解释为 char
,并输出一个字符串而不是数字数组。
如果是 []int
,我可以让它工作,但如果可以避免,我不想分配和复制项目。我能做到吗?
[]byte
将被编码为Base64字符串。
因此,我认为您可能需要使您的结构体实现Marshaler接口,通过实现自己的"数组和切片值编码为JSON数组,除了[]byte编码为base64编码的字符串,nil切片编码为null JSON对象。"
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]}
}
nil
切片处理为null
。 - the system
json.Marshal
。 我的问题不是关于fmt
,而是关于json
库。 - Joe