我们開始须要使用protobuf的原因是,在处理OJ的contest模块的时候,碰到一个问题就是生成contestRank的时候。须要存储非常多信息。

假设我们採用model存储的话,那么一方面兴许假设继续加入其它信息的话改动会灰常麻烦。另一方面就是实现比較复杂。由于对于rank来说,每一条rank的主键首先是UserID,其次存储的基本信息有AC数,题目AC情况,罚时等等,当中题目AC情况又包含以题目ID为主键,属性(是否AC。AC时间。罚时。这道题目的提交统计)。而每一道题目的提交统计又是一个submit实体,这种话。就会有多级嵌套存在。

假设兴许加入的话会很多其它。

另一个原因就是存储也不方便,逻辑构思不是非常明白。须要注意保存各种信息。实现复杂。

protobuf的主要机理是把存储的信息序列化为一个字符串存储。须要信息的时候又能够逆序列化把原始信息还原出来。并且能够实现多级嵌套的属性。所以就尝试使用这个来解决rank的存储(/ □ \)

1:protobuf是什么?

好吧,这个有百科,see see:protocolbuffer是google 的一种数据交换的格式,它独立于语言,独立于平台。google 提供了三种语言的实现:java、c++ 和 python。每一种实现都包括了对应语言的编译器以及库文件。

因为它是一种二进制的格式,比使用
xml 进行数据交换快很多。能够把它用于分布式应用之间的数据通信或者异构环境下的数据交换。

作为一种效率和兼容性都非常优秀的二进制传输数据格式,能够用于诸如网络传输、配置文件、数据存储等诸多领域。

简单地说,就是把某种数据结构的信息,以某种格式保存起来。

主要用于数据存储、传输协议格式等场合。

2:下载安装:

地址:http://code.google.com/p/protobuf/downloads/list

安装:

 tar -xzf protobuf-2.1.0.tar.gz
cd protobuf-2.1.0
./configure
make
make check
make install

3:使用建立

网上有各种样例,比方简单的book的书写等等都能够用来练习操作

首先,首先我们须要编写一个 proto 文件,定义我们程序中须要处理的结构化数据,在 protobuf 的术语中,结构

化数据被称为 Message。比如:

package contest_submit;
message ContestSubmit {
required int32 id = 1;
required int64 timestamp = 2;
}

上述样例中id是int32的,timestamp是int64的。required代表这个是必须的,有点像数据库里的not null,另一

种就是optional。就可以选的。

编译.proto文件,命令例如以下:

protoc -I=./ --python_out=./ ./*.proto

代表的意思是输入的.proto文件在当前文件夹,输出生成的编译文件到当前文件夹,源文件是全部以.proto结尾的文件

网上完整的格式是:

如果您的 proto 文件存放在 $SRC_DIR 以下,您也想把生成的文件放在同一个文件夹下,则能够使用例如以下命令:
protoc -I=$SRC_DIR --python_out=$DST_DIR $SRC_DIR/文件明.proto

编译完毕之后会生成 contest_submit_pb2.py文件(对于上面的样例来说)

当然,protobuf也是能够嵌套的。即能够在一个类中声明还有一个类的对象,比如,rank的实现:

package contest_rank;
message Submit {
optional string status = 1;
optional string date_time = 2;
optional string runID = 3;
} message Problem {
optional int32 problemID = 1;
repeated Submit submit = 2;
optional int32 acindex = 3;
optional int32 totalindex = 4;
optional int32 time = 5;
optional int32 FB = 6;
} message Rank {
optional string userID = 1;
optional string contestID = 2;
repeated Problem problem = 3;
optional int32 penalty = 4;
optional int32 ac = 5;
optional int32 total = 6;
} message ContestRankList {
optional string contestID = 1;
repeated Rank rank = 2;
}

这种话。就能够实现我们前面的需求即多级信息嵌套了。

使用:

首先,要记得加载头文件from 目录.proto import rank_pb2

对于实现rank的存储的时候。首先声明一个rank类,以便兴许对数据的处理

class Rank():
class Problem():
class Submit():
def __init__(self, runID = "", status="", date_time=""):
self.status = status
self.date_time = date_time
self.runID = runID def __init__(self, problemID):
self.problemID = problemID
self.submit_list = []
self.acindex = 0
self.totalindex = 0
self.time = 0
self.FB = 0 def add_submit(self, submit):
self.submit_list.append(submit) def __init__(self, userID, contestID):
self.userID = userID
self.contestID = contestID
self.problem_list = {}
self.ac = 0
self.penalty = 0
self.total = 0

当我们得到rank_list的时候(rank_list是一个存放rank的字典),首先声明一个proto的载体:

contest_rank_list = rank_pb2.ContestRankList()

然后填充信息,比方先把比赛的ID增加,即contest_rank_list.contestID = contestID。

然后依照protobuf的结构,依

次把须要的信息填入。

对于嵌套的实体来说,每次声明的时候能够使用add()方法,比方在一个比赛中,会有多个用户

的rank信息存在。所以依照用户ID来加入信息:

rank_proto = contest_rank_list.rank.add()
rank.load_data_to_proto(rank_proto)

当中的load_data_to_proto是rank类的一个方法。内容就是把信息增加到protobuf里面而已:

def load_data_to_proto(self, rank):
rank.userID = self.userID
rank.contestID = self.contestID
rank.ac = self.ac
rank.penalty = self.penalty
rank.total = self.total for problemID in self.problem_list:
problem = self.problem_list[problemID]
p = rank.problem.add()
p.problemID = problem.problemID
p.acindex = problem.acindex
p.totalindex = problem.totalindex
p.time = problem.time
p.FB = problem.FB
for submit in problem.submit_list:
s = p.submit.add()
s.status = submit.status
s.date_time = submit.date_time
s.runID = submit.runID
return rank

这样我们就把数据存入了我们声明的protobuf里面了。然后就是存储的时候序列化。比方我们存储到SSDB里面即:

ssdb_api.SetContestRankListProto(contestID, contest_rank_list.SerializeToString())

至此。信息已经存入数据库。那么接下来就是怎样读出了:

对于得到我们存储的rank信息来说。即:

 contest_rank_list = ssdb_api.GetContestRankListProto(contest_id)
rank_list = rank_pb2.ContestRankList()
rank_list.ParseFromString(contest_rank_list)

即,再次声明一个protobuf的对象,然后逆序列化。就得到了我们须要的信息。简单吧(/ □ \)。

后面须要的数据。就用父名.子名訪问就可以。

Protocol Buffers的基础说明和使用的更多相关文章

  1. Hadoop基础-Protocol Buffers串行化与反串行化

    Hadoop基础-Protocol Buffers串行化与反串行化 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 我们之前学习过很多种序列化文件格式,比如python中的pickl ...

  2. 使用 Protocol Buffers 代替 JSON 的五个原因

    国内私募机构九鼎控股打造APP,来就送 20元现金领取地址:http://jdb.jiudingcapital.com/phone.html内部邀请码:C8E245J (不写邀请码,没有现金送)国内私 ...

  3. 在centos 6.9下Protocol Buffers数据传输及存储协议的使用(python)

    我们知道Protocol Buffers是Google定义的一种跨语言.跨平台.可扩展的数据传输及存储的协议,因为将字段协议分别放在传输两端,传输数据中只包含数据本身,不需要包含字段说明,所以传输数据 ...

  4. Google Protocol Buffers 快速入门(带生成C#源码的方法)

    Google Protocol Buffers是google出品的一个协议生成工具,特点就是跨平台,效率高,速度快,对我们自己的程序定义和使用私有协议很有帮助. Protocol Buffers入门: ...

  5. 让Web API支持Protocol Buffers

    简介 现在我们Web API项目基本上都是使用的Json作为通信的格式,随着移动互联网的兴起,Web API不仅其他系统可以使用,手机端也可以使用,但是手机端也有相对特殊的地方,网络通信除了wifi, ...

  6. Xml,Json,Hessian,Protocol Buffers序列化对比

    简介 这篇博客主要对Xml,Json,Hessian,Protocol Buffers的序列化和反序列化性能进行对比,Xml和Json的基本概念就不说了. Hessian:Hessian是一个轻量级的 ...

  7. Protocol buffers 介绍

    Protocol buffers和mxl一样在序列化数据结构时很灵活.高效和智能,但是它的优势在于定义文件更小,读取速度更快,使用更加简单.目前protocol buffers支持C++.java和p ...

  8. C#/net 使用Protocol Buffers入门

    Protocol buffers 是一个由谷歌开发的开源的编码机制用于将结构化的数据序列化或者反序列化,被设计成语言以及平台中立,protobuff比xml更简单比json还要紧凑一些,网上有一些关于 ...

  9. java&Protocol Buffers

    ps: Protocol Buffers简称PB PB 安装配置 下载 PB: 在 PB 官网,下载最新版(或者其他版本)PB,这里为了与 Java 项目中的 PB Maven 依赖版本一致,使用 P ...

随机推荐

  1. 原生的ajax请求----(播放托管到爱奇艺上的视频)

    播放视频 $(function(){ //视频播放 $('.play-icon').click(function () { $.ajax({ type:"get", url: &q ...

  2. (2016北京集训十四)【xsy1557】task

    题解: 限制可以看成图状结构,每个任务的对物品数量的影响可以看成权值,只不过这个权值用一个五元组来表示. 那么题意要求的就是最大权闭合子图,网络流经典应用. 代码: #include<algor ...

  3. HDU-1215 七夕节 数论 唯一分解定理 求约数之和

    题目链接:https://cn.vjudge.net/problem/HDU-1215 题意 中文题,自己去看吧,懒得写:) 思路 \[ Ans=\prod \sum p_i^j \] 唯一分解定理 ...

  4. PageUtil

    package cn.com.qmhd.oto.common; import java.io.Serializable; import java.util.List; import org.sprin ...

  5. Docker yum 安装

      [liwm@Eren ~]$ sudo su[root@Eren liwm]# yum install -y docker 已加载插件:fastestmirror, langpacks, prod ...

  6. 现代C++

    C++ 是世界上最常用的编程语言之一. 编写良好的 C++ 程序是快速.高效的. 该语言比其他语言更加灵活,因为你可以使用它来创建各种应用,包括有趣刺激的游戏.高性能科学软件.设备驱动程序.嵌入式程序 ...

  7. JAVA学习第四十六课 — 其它对象API(二)Date类 &amp; Calendar类(重点掌握)

    Date类(重点) 开发时,会时常遇见时间显示的情况,所以必须熟练Date的应用 <span style="font-family:KaiTi_GB2312;font-size:18p ...

  8. Android学习笔记(9):使用XML文件和Java代码控制UI界面

    Android推荐使用XML文件设置UI界面.然后用Java代码控制逻辑部分,这体现了MVC思想. MVC全名是Model View Controller.是模型(model)-视图(view)-控制 ...

  9. angularjs 服务供应商

    <!DOCTYPE HTML> <html ng-app="myApp"> <head> <meta http-equiv="C ...

  10. 反弹木马——本质上就是一个开80端口的CS程序,伪造自己在浏览网页

    反弹端口型木马分析了防火墙的特性后发现:防火墙对于连入的链接往往会进行非常严格的过滤,但是对于连出的链接却疏于防范.于是,与一般的木马相反,反弹端口型木马的服务端(被控制端)使用主动端口,客户端(控制 ...