【实践】给proto的message添加自定义tag

2024-05-14 07:36

本文主要是介绍【实践】给proto的message添加自定义tag,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

背景

通常来说, 使用proto定义message的Field是使用下划线,比如:

# proto定义
message Req {string key_name = 1;
}# 生成的.pb.go 中req的定义type Req struct {state         protoimpl.MessageStatesizeCache     protoimpl.SizeCacheunknownFields protoimpl.UnknownFieldsKeyName            string            `protobuf:"bytes,1,opt,name=key_name,json=keyName,proto3" json:"key_name,omitempty"`

注意: proto的定义,现在默认推荐proto3, 所以此处忽略 proto2中可以给字段声明optional和required。

不过,需要特别注意一下tag中如果添加了omitempty选项,proto2和proto3中对于默认值的处理有所不同。

比如:proto2中对于添加了optional的bool类型,生成的字段是个指针类型,默认值为nil才是empty,false就不是默认值了,而对于proto3中bool的定义,默认值就是false,对于添加omitempty的tag来说就是空了(切记)

从生产的pb.go文件可以看到, 默认生成的json tag是根据字段名的定义来的

可能你会疑惑: protobuf声明的tag中有name=key_name和json=keyName的两者,这应该是用在不同marshal和unMarshal的场景, 比如之前提到的jsonpb库,可以指定marshal的选项: OrigName= true / fasle: 

import ("github.com/gogo/protobuf/jsonpb""encoding/json"
)marshaler := &jsonpb.Marshaler{OrigName:     false,  // 是否按字段的声明(.proto) nameEnumsAsInts:  true,EmitDefaults: true,}# 一个req 示例
req := &Req {KeyName: "123",
}# OrigName=false
out, _ := marshaler.MarshalToString(req) 
t.Logf("out of jsopb:%v\n", out). // {"keyName":"123"}outJson, _ := json.Marshal(req)
t.Logf("out of json:%v\n", string(outJson)) // {"key_name":"123"}# OrigName=true
marshaler = jsonpb.Marshaler{OrigName: true,}out, _ = marshaler.MarshalToString(req)
t.Logf("out of jsopb:%v\n", out). // {"key_name":"123"}
  • OrigName= true:  此时就是按照protobuf声明的tag中name=key_name来处理的
  • OrigName= false:   此时就是按照protobuf声明的tag中json=keyName来处理的

 

到此阶段, pb能生成的基本就是这样的。那如果我想添加自定义的tag呢? 

比如: keyName 映射成 key_real_name 等, 这样的需求应该还不少

添加自定义tag

可以使用这个工具来给Field添加

先安装一下:

go install github.com/favadi/protoc-go-inject-tag@latest

修改对应的proto定义, 然后执行:protoc-go-inject-tag -input="./*.pb.go"

message Req {string key_name = 1; // @gotags: json:"key_real_name" bson:"key_in_monogo" orm:"key_in_orm"
}# 对应的pb生产文件:
type Req struct {state         protoimpl.MessageStatesizeCache     protoimpl.SizeCacheunknownFields protoimpl.UnknownFieldsKeyName string `protobuf:"bytes,1,opt,name=key_name,json=keyName,proto3" json:"key_real_name" bson:"key_in_monogo" orm:"key_in_orm"` // @gotags: json:"key_real_name" bson:"key_in_monogo" orm:"key_in_orm"
}

可以看到,通过@gotags 自定义的tag声明, 生效了。

1. json的tag可以指定新的名称

2. 添加了用于映射mongodb中字段的bson tag

3. 同时, 还添加了orm组件的字段tag 映射名称

是不是很方便, Enjoy ~

小技巧

问题: 如果想要复用proto的message定义, 一般可以直接在proto中引用对应message的定义即可, 那如果我不行新增一个层级呢?

正常使用情况:

message Req {string key_name = 1; // @gotags: json:"key_real_name" bson:"key_in_monogo" orm:"key_in_orm"
}message ReqMeta {string meta = 1;
}// 引用Req和ReqMeta
message ReqAndMeta {Req req = 1;ReqMeta meta = 2;
}

 如果是这样, json格式为:

{"req":{"key_real_name":"123"},"meta":{"meta":"meta123"}}

 

如果这就是你想要的, 那通过在proto中定义一下就可以了。

如果你不想要新增一个req和meta层级呢?

直接在proto中引用定义肯定不行, 但是可以使用golang的特性: 组合

type ReqAndMetaCombine struct {*Req*ReqMeta
}

通过组合多个proto中的message,最后的json格式如下:

{"key_real_name":"123","meta":"meta123"}

确实少了一个层级, it works ~

 

这篇关于【实践】给proto的message添加自定义tag的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/988141

相关文章

SpringBoot通过main方法启动web项目实践

《SpringBoot通过main方法启动web项目实践》SpringBoot通过SpringApplication.run()启动Web项目,自动推断应用类型,加载初始化器与监听器,配置Spring... 目录1. 启动入口:SpringApplication.run()2. SpringApplicat

聊聊springboot中如何自定义消息转换器

《聊聊springboot中如何自定义消息转换器》SpringBoot通过HttpMessageConverter处理HTTP数据转换,支持多种媒体类型,接下来通过本文给大家介绍springboot中... 目录核心接口springboot默认提供的转换器如何自定义消息转换器Spring Boot 中的消息

Java整合Protocol Buffers实现高效数据序列化实践

《Java整合ProtocolBuffers实现高效数据序列化实践》ProtocolBuffers是Google开发的一种语言中立、平台中立、可扩展的结构化数据序列化机制,类似于XML但更小、更快... 目录一、Protocol Buffers简介1.1 什么是Protocol Buffers1.2 Pro

linux安装、更新、卸载anaconda实践

《linux安装、更新、卸载anaconda实践》Anaconda是基于conda的科学计算环境,集成1400+包及依赖,安装需下载脚本、接受协议、设置路径、配置环境变量,更新与卸载通过conda命令... 目录随意找一个目录下载安装脚本检查许可证协议,ENTER就可以安装完毕之后激活anaconda安装更

Android 缓存日志Logcat导出与分析最佳实践

《Android缓存日志Logcat导出与分析最佳实践》本文全面介绍AndroidLogcat缓存日志的导出与分析方法,涵盖按进程、缓冲区类型及日志级别过滤,自动化工具使用,常见问题解决方案和最佳实... 目录android 缓存日志(Logcat)导出与分析全攻略为什么要导出缓存日志?按需过滤导出1. 按

MySQL数据类型与表操作全指南( 从基础到高级实践)

《MySQL数据类型与表操作全指南(从基础到高级实践)》本文详解MySQL数据类型分类(数值、日期/时间、字符串)及表操作(创建、修改、维护),涵盖优化技巧如数据类型选择、备份、分区,强调规范设计与... 目录mysql数据类型详解数值类型日期时间类型字符串类型表操作全解析创建表修改表结构添加列修改列删除列

Python自定义异常的全面指南(入门到实践)

《Python自定义异常的全面指南(入门到实践)》想象你正在开发一个银行系统,用户转账时余额不足,如果直接抛出ValueError,调用方很难区分是金额格式错误还是余额不足,这正是Python自定义异... 目录引言:为什么需要自定义异常一、异常基础:先搞懂python的异常体系1.1 异常是什么?1.2

深入解析Java NIO在高并发场景下的性能优化实践指南

《深入解析JavaNIO在高并发场景下的性能优化实践指南》随着互联网业务不断演进,对高并发、低延时网络服务的需求日益增长,本文将深入解析JavaNIO在高并发场景下的性能优化方法,希望对大家有所帮助... 目录简介一、技术背景与应用场景二、核心原理深入分析2.1 Selector多路复用2.2 Buffer

Linux中的自定义协议+序列反序列化用法

《Linux中的自定义协议+序列反序列化用法》文章探讨网络程序在应用层的实现,涉及TCP协议的数据传输机制、结构化数据的序列化与反序列化方法,以及通过JSON和自定义协议构建网络计算器的思路,强调分层... 目录一,再次理解协议二,序列化和反序列化三,实现网络计算器3.1 日志文件3.2Socket.hpp

C语言自定义类型之联合和枚举解读

《C语言自定义类型之联合和枚举解读》联合体共享内存,大小由最大成员决定,遵循对齐规则;枚举类型列举可能值,提升可读性和类型安全性,两者在C语言中用于优化内存和程序效率... 目录一、联合体1.1 联合体类型的声明1.2 联合体的特点1.2.1 特点11.2.2 特点21.2.3 特点31.3 联合体的大小1