最近尝试用 gin 写一个个人项目,有个问题就是,因为 golang 里用大小写控制访问权限,所以我定义的 struct 的属性名都是大写的,这样以 json 格式返回后 key 也都是首字母大写的 PascalCase 风格,但我需要 cameCase 风格,有什么办法统一处理吗?
我目前知道可以在定义 struct 的时候加json的 tag,但是一个有点麻烦,另一个就是对于一些第三方依赖包里定义的 struct 就没有办法了,有什么优雅的办法吗?
最近尝试用 gin 写一个个人项目,有个问题就是,因为 golang 里用大小写控制访问权限,所以我定义的 struct 的属性名都是大写的,这样以 json 格式返回后 key 也都是首字母大写的 PascalCase 风格,但我需要 cameCase 风格,有什么办法统一处理吗?
我目前知道可以在定义 struct 的时候加json的 tag,但是一个有点麻烦,另一个就是对于一些第三方依赖包里定义的 struct 就没有办法了,有什么优雅的办法吗?
1
Hanggi Jun 12, 2020
请认真阅读文档:
https://github.com/gin-gonic/gin#model-binding-and-validation 如果遇到路由冲突: https://hanggi.me/post/golang/wildcard-conflict/ 当然好好设计针对 httprouter 的路由规则才是正解。 |
2
beiping96 Jun 12, 2020
`encoding/json`最优雅
|
3
dilu Jun 12, 2020
u1s1,个人项目用 php 一把梭不香嘛 或者你擅长的语言也可以噻
|
6
Fitz Jun 12, 2020
加个 tag 有什么麻烦的, 第三方的你只能自己定义一个一样的结构体,把数据拷过来了
|
7
eslizn Jun 12, 2020
给第三方 struct 定义别名,并给这个别名实现 Marshaler 接口
|
8
freemoon Jun 12, 2020
默认就是和属性名保持一致,没毛病,要驼峰加 tag,这个做法没有问题,你的需求是个性化的。
|
9
freemoon Jun 12, 2020
要么自己实现对应接口,那个成本更高
|
10
Hanggi Jun 12, 2020 via iPhone
@lancelock 原理是一样的,静态语言就是要把 json 绑定到结构体里。不要想着动态解析它,用不来可以用 nodejs,现在 typescript 开发也都要绑定结构体,一样的。
|
11
lancelock OP 按我的想法,这个可以用一个后置的中间件,handlle 结束后做统一处理,可是没找到现成的,我试试自己写一个吧
|
12
qloog Jun 12, 2020
一般的做法是在 struct 的 tag 进行处理,添加 json,里面随便定义,下划线或者驼峰方式都可以。
|
13
AlphaTr Jun 12, 2020 via iPhone
对 第三方 struct 应该可以用 reflect 处理
|
15
whoami9894 Jun 12, 2020
笑死了 #1 #10,一本正经的答非所问
可以找一个自动生成 tag 的结构体生成器,第三方包的话就把字段反射解析出来 |
16
dai875939260 Jun 12, 2020
https://github.com/fatih/gomodifytags 修改 tag,第三方的包的可以自己再建一个结构体,然后用 mapstructure 转换?
|
17
wangyzj Jun 12, 2020
go 这个东西不就是这样
你都习惯别的了还纠结啥 tag |
18
scnace Jun 12, 2020 via Android
自己实现一个自用 Marshaller ?
|
19
Kisesy Jun 12, 2020
只需要实现 Marshaler 接口并包裹一下(这篇文章中用正则表达式来处理比较低效,需要优化)
https://www.cnblogs.com/chenqionghe/p/13067596.html |
20
mengzhuo Jun 12, 2020
https://json.to-go.online
|
21
Hanggi Jun 12, 2020
@lancelock 我说得很清楚了,Go 是静态语言,不要想着什么 marshal 之类的优雅解决。
你要做的就是定义一个 response 结构体,把你要返回的数据 mapping 上去。 你可以用一些工具简化这个操作,但是结果都是一样的。 如果你觉得麻烦,不想这样,可以考虑使用 Nodejs 这种动态语言。 |
23
blessyou Jun 12, 2020
那看起来还是转换风格比较 easy 。
|
24
sunxiansong Jun 12, 2020
VSCode go 插件有个命令:Go: Add Tags To Struct Fields
不过缺点是支持 snake_case,不知道在哪里改 好在我找到 VSCode 有个插件 change-case ,可以方便的在不认风格间切换 |
25
yxlimo Jun 12, 2020 via iPhone
我觉得问题的点在于,你为什么要依赖第三方包的 struct 生成 response 。这样子是不可控的
|
26
Hanggi Jun 12, 2020
@lancelock 怪心累的,听不懂人话呢,只说最后一次。
看你问这个问题就知道你没怎么用过 go 语言。你以为你是这世界上第一个有这种需求的人吗? go 语言发展了 10 年为什么找不到你要的所谓的第三方依赖包,因为 go 语言不提倡这么做。 像 #13 楼说的 reflect 其实就是一种动态判断类型的方法,但是,不要这么做,原因自己查。 不要什么后置中间件统一处理,如果你非要这么做可能 go 语言不适合你。 go 语言 json struct 可以嵌套你知道吧?你要做的就是定义好结构体,加好 tag,组合使用他们。 如果这样还看不懂,你也不用回了。 |
27
labulaka521 Jun 12, 2020 via iPhone
定义 tag ?
|
28
iceiceice Jun 12, 2020
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 } /*************************************** 其他方法 ***************************************/ // 驼峰式写法转为下划线写法 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() } // 下划线写法转为驼峰写法 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 "" } // 内嵌 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("*****内存不够了!******") } }() b.WriteString(s) return b } func TestJsonCamelCase_MarshalJSON(t *testing.T) { type Person struct { HelloWold string LightWeightBaby string } var a = Person{HelloWold: "xxx", LightWeightBaby: "muscle"} res, _ := json.Marshal(JsonCamelCase{a}) fmt.Printf("%s", res) } |
29
BlackBerry999 Jun 12, 2020
@iceiceice 秀!
|