protobuf与json互相转换
Java
http://code.google.com/p/protobuf-java-format/
maven
<dependency>
<groupId>com.googlecode.protobuf-java-format</groupId>
<artifactId>protobuf-java-format</artifactId>
<version>1.2</version>
</dependency>
从protobuf转json
Message someProto =SomeProto.getDefaultInstance();
String jsonFormat =JsonFormat.printToString(someProto);
从json转protobuf
Message.Builder builder =SomeProto.newBuilder();
String jsonFormat = _load json document from a source_;
JsonFormat.merge(jsonFormat, builder);
C++
https://github.com/shramov/json2pb
Python
https://github.com/NextTuesday/py-pb-converters
导入模块pbjson.py即可使用。
ps. 原始模块的pb2json函数会自动过滤protobuf中字段值为空的数据。根据需要可注释掉过滤代码。
pbjson .py:
| import simplejson | |
|
from google.protobuf.descriptor import FieldDescriptor as FD |
|
|
class ConvertException(Exception): |
|
| pass | |
|
def dict2pb(cls, adict, strict=False): |
|
| """ | |
| Takes a class representing the ProtoBuf Message and fills it with data from | |
| the dict. | |
| """ | |
|
obj = cls() |
|
|
for field in obj.DESCRIPTOR.fields: |
|
|
if not field.label == field.LABEL_REQUIRED: |
|
| continue | |
|
if not field.has_default_value: |
|
| continue | |
|
if not field.name in adict: |
|
| raise ConvertException('Field "%s" missing from descriptor dictionary.' | |
| % field.name) | |
|
field_names = set([field.name for field in obj.DESCRIPTOR.fields]) |
|
| if strict: | |
|
for key in adict.keys(): |
|
|
if key not in field_names: |
|
| raise ConvertException( | |
|
'Key "%s" can not be mapped to field in %s class.' |
|
|
% (key, type(obj))) |
|
|
for field in obj.DESCRIPTOR.fields: |
|
|
if not field.name in adict: |
|
| continue | |
| msg_type = field.message_type | |
|
if field.label == FD.LABEL_REPEATED: |
|
|
if field.type == FD.TYPE_MESSAGE: |
|
|
for sub_dict in adict[field.name]: |
|
|
item = getattr(obj, field.name).add() |
|
| item.CopyFrom(dict2pb(msg_type._concrete_class, sub_dict)) | |
| else: | |
| map(getattr(obj, field.name).append, adict[field.name]) | |
| else: | |
|
if field.type == FD.TYPE_MESSAGE: |
|
| value = dict2pb(msg_type._concrete_class, adict[field.name]) | |
| getattr(obj, field.name).CopyFrom(value) | |
| else: | |
| setattr(obj, field.name, adict[field.name]) | |
| return obj | |
|
def pb2dict(obj): |
|
| """ | |
| Takes a ProtoBuf Message obj and convertes it to a dict. | |
| """ | |
| adict = {} | |
|
if not obj.IsInitialized(): |
|
|
return None |
|
|
for field in obj.DESCRIPTOR.fields: |
|
|
if not getattr(obj, field.name): |
|
| continue | |
|
if not field.label == FD.LABEL_REPEATED: |
|
|
if not field.type == FD.TYPE_MESSAGE: |
|
|
adict[field.name] = getattr(obj, field.name) |
|
| else: | |
| value = pb2dict(getattr(obj, field.name)) | |
| if value: | |
| adict[field.name] = value | |
| else: | |
|
if field.type == FD.TYPE_MESSAGE: |
|
| adict[field.name] = \ | |
|
[pb2dict(v) for v in getattr(obj, field.name)] |
|
| else: | |
|
adict[field.name] = [v for v in getattr(obj, field.name)] |
|
| return adict | |
|
def json2pb(cls, json, strict=False): |
|
| """ | |
| Takes a class representing the Protobuf Message and fills it with data from | |
| the json string. | |
| """ | |
| return dict2pb(cls, simplejson.loads(json), strict) | |
|
def pb2json(obj): |
|
| """ | |
| Takes a ProtoBuf Message obj and convertes it to a json string. | |
| """ | |
|
return simplejson.dumps(pb2dict(obj), sort_keys=True, indent=4) |
json2pb
如果测试脚本用Python来写的话,一般的做法是用的Python调用json模块中函数解析json文件,然后新建protobuf对象,依次填写各字段,然后发送出去。
当然如果愿意造上面的轮子也没关系,如果想偷懒也是可行的,pbjson.py脚本就为你干这事提供了温床
1、造json文件,内容如下:
#vi testjson2pb.json
{
"name":"scq",
"age":30,
"work_unit":"taobao",
"class_mate":[
{
"name":"jim",
"age":30
}
]
}
2、造protobuf文件,内容如下:
#vi testjson2pb.proto
//class mate
package json2pb;
message PersonInfo {
//my name
required string name = 1;
//my age
optional int32 age = 2;
//my work unit
optional string work_unit = 3;
message ClassMate {
//name
optional string name = 1;
//age
optional uint32 age = 2;
}
repeated ClassMate class_mate = 4;
}
3、生成protobuf对应python文件:
#protoc -I=. --python_out=. testjson2pb.proto
testjson2pb_pb2.py
4、编写测试例子,具体如下:
#vi testjson2pb.py
-------------------------------------------------------------------------------------------------
#!/usr/bin/env python
# coding=utf-8
import sys
import logging
import os
import time
import string
import pbjson
import simplejson
import testjson2pb_pb2
def main():
f = file("testjson2pb.json")
s = simplejson.load(f)
f.close
t = pbjson.dict2pb(testjson2pb_pb2.PersonInfo, s)
print t
if __name__ == "__main__":
main()
5、执行脚本,结果如下:
#python testjson2pb.py

是不是很简单啊,自从有了这个脚本,再用不用担心人肉从json生成protobuf的问题了
。
protobuf与json互相转换的更多相关文章
- protobuf与json转换
protobuf对象不能直接使用jsonlib去转,因为protobuf生成的对象的get方法返回的类型有byte[],而只有String类型可以作为json的key,protobuf提供方法进行转换 ...
- protobuf转json
方法介绍 protobuf的介绍在这里就不详细介绍了,主要是俺也是刚接触,感兴趣的同学可以去搜索相关博客或者直接去看源码以及google的官方文档(官方文档表示很吃力)或者去这个网站:https:// ...
- protobuf与json相互转换的方法
google的protobuf对象转json,不能直接使用FastJson之类的工具进行转换,原因是protobuf生成对象的get方法,返回的类型有byte[],而只有String类型可以作为jso ...
- json日期转换
//调用 ChangeDateFormat(CreatTime) //json日期转换 function ChangeDateFormat(jsondate) { jsondate = jsondat ...
- java中Array/List/Map/Object与Json互相转换详解
http://blog.csdn.net/xiaomu709421487/article/details/51456705 JSON(JavaScript Object Notation): 是一种轻 ...
- Json格式转换
验证Json格式可以进入 http://json.cn/ json简单说就是javascript中的对象和数组,所以这两种结构就是对象和数组两种结构,通过这两种结构可以表示各种复杂的结构1.对象:对象 ...
- JSON.stringify转换Date不正确的解決方法
JSON.stringify转换Date不正确的原因:国际时区(UTC)和中国时区(GMT)的原因,东八区+8等于国际时区. 解决方法,重新Es5的Date.prototype.toJSON方法,代码 ...
- 使用js进行string和json之间转换的方法
在数据传输过种中,json是以文本,即字符串的形式传递,字符串形似Json对象: var str1 = '{ "name": "Amy", "sex& ...
- 前台 JSON对象转换成字符串 相互转换 的几种方式
在最近的工作中,使用到JSON进行数据的传递,特别是从前端传递到后台,前台可以直接采用ajax的data函数,按json格式传递,后台Request即可,但有的时候,需要传递多个参数,后台使用requ ...
随机推荐
- url 取出文件扩展名
/**url 取出文件扩展名 *///方法一function getExt1($url) { $arr = parse_url($url); $file = basename($arr[' ...
- iOS UITextView 根据输入text自适应高度
转载自:http://www.cnblogs.com/tmf-4838/p/5380495.html #import "ViewController.h" @interface V ...
- 一些常见warning的原因和解决方法
在入职三周后,终于赶齐了接手项目落下两个月的项目,有了一些自己的空闲时间对项目进行整理.主要整理包括类目的整合,从原来一个系统文件夹下几百个文件整改为以MVC设计思想为原则的分文件夹整理类目,井然有序 ...
- px和em,rem的区别
任意浏览器的默认字体高都是16px. px: 像素(Pixel) , 计算机屏幕上的一个点.固定大小:不方便维护: em:相对于当前对象内 (父元素) 文本的字体尺寸.如当前对行内文本的字体尺寸未被人 ...
- Jquery实现表格的分页
功能描述 该分页功能不是一次将所有记录都加载出来,然后在点击按钮翻页的时候,通过设置每一条记录的display属性展示或隐藏实现分页的效果.由于后台的获取数据的接口已实现分页的功能,所以在点击翻页按钮 ...
- webservice简介及客户端搭建
在环境变量中,CLASSPATH添加D:\apache-cxf-2.4.2\lib 新建CXF_HOME D:\apache-cxf-2.4.2 在PATH中添加 D:\apache-cxf- ...
- linux安装文件命令
tar -zxvf apache-tomcat.tar.gz -C /home/poka 注:安装tar.gz的安装包 设置系统自动启动tomcat 切换到root用户,执行命令 #chkconfig ...
- ios 基础数据类型
1:NSString http://blog.csdn.net/likendsl/article/details/7417878 http://xys289187120.blog.51cto.com/ ...
- VBS 批量修改多个文件夹下的文字命名
Function FilesTree(sPath) Set oFso = CreateObject("Scripting.FileSystemObject") ...
- iOS中date和string的相互转换
必须知道的内容 G: 公元时代,例如AD公元 yy: 年的后2位 yyyy: 完整年 MM: 月,显示为1-12 MMM: 月,显示为英文月份简写,如 Jan ...