一、需求
golang默認的結(jié)構(gòu)體json轉(zhuǎn)碼出來,都是根據(jù)字段名生成的大寫駝峰格式,但是一般我們最常用的json格式是小寫駝峰或者小寫下劃線,因此,我們非常需要一個統(tǒng)一的方法去轉(zhuǎn)換,而不想挨個寫json標(biāo)簽,例如
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
package main import ( "encoding/json" "fmt" ) func main() { type Person struct { HelloWold string LightWeightBaby string } var a = Person{HelloWold: "chenqionghe", LightWeightBaby: "muscle"} res, _ := json.Marshal(a) fmt.Printf("%s", res) } |
運行結(jié)果
{"HelloWold":"chenqionghe","LightWeightBaby":"muscle"}
輸出來的json結(jié)構(gòu)是大寫駝峰的,肯定非常別扭的,當(dāng)然 ,我們通過設(shè)定json標(biāo)簽來設(shè)定輸出的json字段名,例如
1
2
3
4
|
type Person struct { HelloWold string `json:"hello_wold"` LightWeightBaby string `json:"light_weight_baby"` } |
但是如果字段特別多,需要挨個設(shè)置也太麻煩了。
二、實現(xiàn)
Golang 的標(biāo)準(zhǔn) Json 在處理各種數(shù)據(jù)類型是都是調(diào)用其類型接口UnmarshalJSON解碼和MarshalJSON編碼進行轉(zhuǎn)換的,所以我們可以封裝一個統(tǒng)一轉(zhuǎn)換下劃線的json結(jié)構(gòu)體或統(tǒng)一轉(zhuǎn)換駝峰的json結(jié)構(gòu)體,并實現(xiàn)MarshalJSON方法,就可以達到目的。
實現(xiàn)如下
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
|
package jsonconv import ( "bytes" "encoding/json" "log" "regexp" "strconv" "strings" "unicode" ) /*************************************** 下劃線json ***************************************/ type JsonSnakeCase struct { Value interface{} } func (c JsonSnakeCase) MarshalJSON() ([]byte, error) { // Regexp definitions var keyMatchRegex = regexp.MustCompile(`\"(\w+)\":`) var wordBarrierRegex = regexp.MustCompile(`(\w)([A-Z])`) marshalled, err := json.Marshal(c.Value) converted := keyMatchRegex.ReplaceAllFunc( marshalled, func(match []byte) []byte { return bytes.ToLower(wordBarrierRegex.ReplaceAll( match, []byte(`${1}_${2}`), )) }, ) return converted, err } /*************************************** 駝峰json ***************************************/ type JsonCamelCase struct { Value interface{} } func (c JsonCamelCase) MarshalJSON() ([]byte, error) { var keyMatchRegex = regexp.MustCompile(`\"(\w+)\":`) marshalled, err := json.Marshal(c.Value) converted := keyMatchRegex.ReplaceAllFunc( marshalled, func(match []byte) []byte { matchStr := string(match) key := matchStr[1 : len(matchStr)-2] resKey := Lcfirst(Case2Camel(key)) return []byte(`"` + resKey + `":`) }, ) return converted, err } /*************************************** 其他方法 ***************************************/ // 駝峰式寫法轉(zhuǎn)為下劃線寫法 func Camel2Case(name string) string { buffer := NewBuffer() for i, r := range name { if unicode.IsUpper(r) { if i != 0 { buffer.Append('_') } buffer.Append(unicode.ToLower(r)) } else { buffer.Append(r) } } return buffer.String() } // 下劃線寫法轉(zhuǎn)為駝峰寫法 func Case2Camel(name string) string { name = strings.Replace(name, "_", " ", -1) name = strings.Title(name) return strings.Replace(name, " ", "", -1) } // 首字母大寫 func Ucfirst(str string) string { for i, v := range str { return string(unicode.ToUpper(v)) + str[i+1:] } return "" } // 首字母小寫 func Lcfirst(str string) string { for i, v := range str { return string(unicode.ToLower(v)) + str[i+1:] } return "" } // 內(nèi)嵌bytes.Buffer,支持連寫 type Buffer struct { *bytes.Buffer } func NewBuffer() *Buffer { return &Buffer{Buffer: new(bytes.Buffer)} } func (b *Buffer) Append(i interface{}) *Buffer { switch val := i.(type) { case int: b.append(strconv.Itoa(val)) case int64: b.append(strconv.FormatInt(val, 10)) case uint: b.append(strconv.FormatUint(uint64(val), 10)) case uint64: b.append(strconv.FormatUint(val, 10)) case string: b.append(val) case []byte: b.Write(val) case rune: b.WriteRune(val) } return b } func (b *Buffer) append(s string) *Buffer { defer func() { if err := recover(); err != nil { log.Println("*****內(nèi)存不夠了!******") } }() b.WriteString(s) return b } |
三、使用
JsonSnakeCase統(tǒng)一轉(zhuǎn)下劃線json
使用jsonconv.JsonSnakeCase包裹一下要輸出json的對象即可
1
2
3
4
5
6
7
8
9
|
func main() { type Person struct { HelloWold string LightWeightBaby string } var a = Person{HelloWold: "chenqionghe", LightWeightBaby: "muscle"} res, _ := json.Marshal(jsonconv.JsonSnakeCase{a}) fmt.Printf("%s", res) } |
輸出如下
{"hello_wold":"chenqionghe","light_weight_baby":"muscle"}
JsonCamelCase統(tǒng)一轉(zhuǎn)駝峰json
已經(jīng)指定了下劃線標(biāo)簽的結(jié)構(gòu)體,我們也可以統(tǒng)一轉(zhuǎn)為駝峰的json
1
2
3
4
5
6
7
8
9
|
func main() { type Person struct { HelloWold string `json:"hello_wold"` LightWeightBaby string `json:"light_weight_baby"` } var a = Person{HelloWold: "chenqionghe", LightWeightBaby: "muscle"} res, _ := json.Marshal(jsonconv.JsonCamelCase{a}) fmt.Printf("%s", res) } |
輸出如下
{"helloWold":"chenqionghe","lightWeightBaby":"muscle"}
非常方便的解決了json統(tǒng)一轉(zhuǎn)碼格式的需求
到此這篇關(guān)于Go語言json編碼駝峰轉(zhuǎn)下劃線、下劃線轉(zhuǎn)駝峰的實現(xiàn)的文章就介紹到這了,更多相關(guān)Go 駝峰轉(zhuǎn)下劃線、下劃線轉(zhuǎn)駝峰內(nèi)容請搜索服務(wù)器之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持服務(wù)器之家!
原文鏈接:https://www.cnblogs.com/chenqionghe/p/13067596.html