亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > C++ > 正文

protobuf C++ 使用示例

2019-11-08 00:49:01
字體:
來源:轉載
供稿:網友

1、在.PRoto文件中定義消息格式

2、使用protobuf編譯器

3、使用c++ api來讀寫消息

 

0、為何使用protobuf?

 

1、原始內存數據結構,可以以二進制方式sent/saved.這種方式需要相同的內存布局和字節序。

2、以ad-hoc方式將數據項編碼成一個簡單字符串----比如,將4個int類型編碼成"12:3:-23:67"。這種方式簡靈活。適用于簡單數據。

3、將數據序列化為xml。這種方式很流行,因為xml可讀性好,編碼解碼方便,性能也好。僅僅XML dom樹比較復雜。

 

protobuf可以很好的解決上述問題。你編寫一個.proto文件來描述數據結構。protobuf編譯器使用它創建一個類,使用二進制方式自動編碼/解碼該數據結構。生成的類提供getter/setter方法。

 

最重要的是,protobuf支持在此基礎上進行格式擴展。

 

示例

 

1、定義協議格式

 

package tutorial;  message Person {

   required string name = 1;

   required int32 id = 2;

   optional string email= 3;

  

    enum PhoneType {

        MOBILE = 0;

        HOME = 1;

        WORK = 2;

   }

 

   message PhoneNumber {

        required string number = 1;

        optional PhoneType type = 2 [default = HOME];  

   }

 

   repeated PhoneNumber phone= 4;

 

message AddressBook {

   repeated Personperson = 1;

}

 

 

該結構與c++或java很像.

 

.proto文件以包聲明開始,防止名字沖突。

簡單類型:bool, int32, float, double, string.

其它類型:如上述的Person, PhoneNumber

 

類型可以嵌套。

“=1”, “=2”標識唯一“tag”.tag數1-15需要至少一個字節。

 

required: 必須設置它的值

optional: 可以設置,也可以不設置它的值

repeated: 可以認為是動態分配的數組

google工程師認為使用required威害更大,他們更喜歡使用optional, repeated.

 

 

2、編譯你的協議

 

運行protoc 來生成c++文件:

protoc -I=$SRC_DIR --cpp_out=$DST_DIR$SRC_DIR/addressbook.proto

生成的文件為:

addressbook.pb.h, 

addressbook.pb.cc

 

3、protobuf API

 

生成的文件中有如下方法:

// name

  inline bool has_name() const;

  inline voidclear_name();

  inline const ::std::string& name() const;

  inline void set_name(const ::std::string& value);

  inline void set_name(const char* value);

  inline ::std::string*mutable_name();

 

  // id

  inline bool has_id() const;

  inline void clear_id();

  inline int32_t id() const;

  inline void set_id(int32_t value);

 

  // email

  inline bool has_email() const;

  inline voidclear_email();

  inline const ::std::string& email() const;

  inline void set_email(const ::std::string& value);

  inline void set_email(const char* value);

  inline ::std::string* mutable_email();

 

  // phone

  inline intphone_size() const;

  inline voidclear_phone();

  inline const ::google::protobuf::RepeatedPtrField< ::tutorial::Person_PhoneNumber >& phone() const;

  inline ::google::protobuf::RepeatedPtrField< ::tutorial::Person_PhoneNumber >*mutable_phone();

  inline const ::tutorial::Person_PhoneNumber& phone(int index) const;

  inline ::tutorial::Person_PhoneNumber*mutable_phone(int index);

  inline ::tutorial::Person_PhoneNumber* add_phone();

4、枚舉與嵌套類

 

生成的代碼包含一個PhoneType枚舉。Person::PhoneType, Person:MOBILE,Person::HOME, Person:WORK.

 

編譯器生成的嵌套類稱為Person::PhoneNumber. 實際生成類為Person_PhoneNumber.

 

5、標準方法

 

bool IsInitialized() const:               確認required字段是否被設置

string DebugString() const:               返回消息的可讀表示,用于調試

void CopyFrom(const Person& from):         使用給定消息值copy

void Clear():                             清除所有元素為空狀態

 

6、解析與序列化

 

bool SerializeToString(string* output) const:        序列化消息,將存儲字節的以string方式輸出。注意字節是二進制,而非文本;

bool ParseFromString(const string& data):            解析給定的string     

bool SerializeToOstream(ostream* output) const:      寫消息給定的c++  ostream中

bool ParseFromIstream(istream* input):              從給定的c++ istream中解析出消息

7、protobuf和 oo設計

不要繼承生成類并在此基礎上添加相應的行為

 

8、寫消息

 

示例:它從一個文件中讀取AddressBook,基于io添加一個新的Person,并將新的AddressBook寫回文件。

#include <iostream>

#include <fstream>

#include <string>

#include"addressbook.pb.h"

using namespace std;

 

// Thisfunction fills in a Person message based on user input.

void PromptForAddress(tutorial::Person* person) {

 cout << "Enter person ID number: ";

  int id;

  cin>> id;

  person->set_id(id);

  cin.ignore(256, '/n');

 

 cout << "Enter name: ";

 getline(cin, *person->mutable_name());

 

 cout << "Enter email address (blank for none):";

  string email;

 getline(cin, email);

  if (!email.empty()) {

   person->set_email(email);

  }

 

  while (true) {

   cout << "Enter a phone number (or leave blank tofinish): ";

   string number;

   getline(cin, number);

   if (number.empty()) {

     break;

   }

 

   tutorial::Person::PhoneNumber* phone_number = person->add_phone();

   phone_number->set_number(number);

 

   cout << "Is this a mobile, home, or work phone?";

   string type;

   getline(cin, type);

   if (type == "mobile") {

     phone_number->set_type(tutorial::Person::MOBILE);

   } else if (type == "home") {

     phone_number->set_type(tutorial::Person::HOME);

   } else if (type == "work") {

     phone_number->set_type(tutorial::Person::WORK);

   } else {

     cout << "Unknownphone type.  Using default." << endl;

   }

  }

}

 

// Mainfunction:  Reads the entire address book from a file,

//  adds one person based on user input, then writes it back out to the same

//  file.

int main(int argc, char* argv[]) {

  // Verifythat the version of the library that we linked against is

  //compatible with the version of the headers we compiled against.

  GOOGLE_PROTOBUF_VERIFY_VERSION;

 

  if (argc != 2) {

   cerr << "Usage:  " << argv[0] << "ADDRESS_BOOK_FILE" << endl;

   return -1;

  }

 

  tutorial::AddressBook address_book;

 

  {

   // Read the existing address book.

   fstream input(argv[1], ios::in | ios::binary);

   if (!input) {

     cout << argv[1] << ":File not found.  Creating a new file." << endl;

   } else if (!address_book.ParseFromIstream(&input)) {

     cerr << "Failedto parse address book." << endl;

     return -1;

   }

  }

 

  // Add anaddress.

  PromptForAddress(address_book.add_person());

 

  {

   // Write the new address book back to disk.

   fstream output(argv[1], ios::out | ios::trunc | ios::binary);

   if (!address_book.SerializeToOstream(&output)) {

     cerr << "Failedto write address book." << endl;

     return -1;

   }

  }

 

  //Optional:  Delete all global objects allocated by libprotobuf.

  google::protobuf::ShutdownProtobufLibrary();

 

  return 0;

}

注意使用GOOGLE_PROTOBUF_VERIFY_VERSION宏。每一個.pb.cc文件在啟動時都將自動調用該宏。

 

注意在程序結尾處調用ShutdownProtobufLibrary()。

 

9、讀消息 

#include <iostream>

#include <fstream>

#include <string>

#include"addressbook.pb.h"

using namespace std;

 

//Iterates though all people in the AddressBook and prints info about them.

void ListPeople(const tutorial::AddressBook& address_book) {

  for (int i = 0; i < address_book.person_size(); i++) {

   const tutorial::Person& person = address_book.person(i);

 

   cout << "Person ID: " << person.id() << endl;

   cout << "  Name: " << person.name() << endl;

   if (person.has_email()) {

     cout << " E-mail address: " << person.email() << endl;

   }

 

   for (int j = 0; j < person.phone_size(); j++) {

     const tutorial::Person::PhoneNumber& phone_number = person.phone(j);

 

     switch (phone_number.type()) {

       case tutorial::Person::MOBILE:

         cout << " Mobile phone #: ";

         break;

       case tutorial::Person::HOME:

         cout << " Home phone #: ";

         break;

       case tutorial::Person::WORK:

         cout << " Work phone #: ";

         break;

     }

     cout << phone_number.number() << endl;

   }

  }

}

 

// Mainfunction:  Reads the entire address book from a file and prints all

//  the information inside.

int main(int argc, char* argv[]) {

  // Verifythat the version of the library that we linked against is

  //compatible with the version of the headers we compiled against.

  GOOGLE_PROTOBUF_VERIFY_VERSION;

 

  if (argc != 2) {

   cerr << "Usage:  " << argv[0] << "ADDRESS_BOOK_FILE" << endl;

   return -1;

  }

 

  tutorial::AddressBook address_book;

 

  {

   // Read the existing address book.

   fstream input(argv[1], ios::in | ios::binary);

   if (!address_book.ParseFromIstream(&input)) {

     cerr << "Failedto parse address book." << endl;

     return -1;

   }

  }

 

  ListPeople(address_book);

 

  //Optional:  Delete all global objects allocated by libprotobuf.

  google::protobuf::ShutdownProtobufLibrary();

 

  return 0;

}

10、擴展protobuf

 

如果希望向后兼容,必須遵循:

a、不必更改tag數

b、不必添加或刪除任何required字段

c、可以刪除optional或repeated字段

d、可以添加新的optional或repeated字段,但你必須使用新的tag數。

 

11、優化

c++的protobuf庫,已經極大地優化了。合理使用可以改善性能。

a、如果可能,復用message對象。

b、關于多線程的內存分配器

 

12、高級用法

 

protobuf的消息類的一個關鍵特性是,反射(reflection)??梢允褂脁ml或json來實現。參考。

 

 

================================================================

常見問題:

1、undefined reference to`pthread_once' 

使用-lpthread:

 

2、error while loading shared libraries:libprotobuf.so.7: cannot open shared object file: No such file or directory

使用-Wl,-Bstatic -lprotobuf -Wl,-Bdynamic-lpthread


發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表

圖片精選

亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
成人国产精品久久久久久亚洲| 亚洲美女激情视频| 3344国产精品免费看| 欧美激情啊啊啊| 日韩美女在线观看| 精品国产一区久久久| www国产精品视频| 黑人欧美xxxx| 91中文字幕在线| 久久亚洲精品视频| 日韩激情第一页| 国产综合在线视频| 久久九九国产精品怡红院| 国产日韩专区在线| 91精品国产综合久久久久久久久| 97激碰免费视频| 成人黄色大片在线免费观看| 亚洲精品videossex少妇| 成人黄色av网站| 亚洲男人第一av网站| 欧美一区二区三区艳史| 日韩黄色在线免费观看| 日韩网站免费观看高清| 欧美日韩亚洲精品内裤| 国产精品盗摄久久久| 色噜噜亚洲精品中文字幕| 国产精品激情av在线播放| 亚洲欧洲日产国码av系列天堂| 欧美极品少妇xxxxⅹ裸体艺术| 欧美怡红院视频一区二区三区| 久久激情视频久久| 亚洲无限乱码一二三四麻| 国产日韩精品在线| 成人免费视频xnxx.com| 国产精品444| 久久久久国产精品www| 亚洲精品黄网在线观看| 精品国偷自产在线视频| 亚洲free性xxxx护士白浆| 欧美激情一区二区三区久久久| 国产成人一区二区三区小说| 日韩在线视频一区| 精品国产一区二区三区四区在线观看| 久久99热精品这里久久精品| 成人国产亚洲精品a区天堂华泰| 欧美视频专区一二在线观看| 亚洲国产日韩欧美在线99| 91高清免费在线观看| 一区二区三欧美| 九九精品视频在线| 中文字幕在线观看日韩| 91精品在线播放| 欧美激情视频在线免费观看 欧美视频免费一| 欧美激情免费看| 欧美日本精品在线| 97成人精品区在线播放| 日韩中文在线中文网在线观看| 国产精品欧美久久久| 成人国产精品免费视频| 成人两性免费视频| 国内精久久久久久久久久人| 狠狠色噜噜狠狠狠狠97| 亚洲成人免费网站| 亚洲成人国产精品| 欧美性videos高清精品| 久久精品亚洲国产| 在线免费观看羞羞视频一区二区| 色偷偷88888欧美精品久久久| 91网站在线免费观看| 国外色69视频在线观看| 亚洲小视频在线观看| 在线日韩欧美视频| 日韩中文字幕第一页| 精品美女久久久久久免费| 777午夜精品福利在线观看| 欧美激情一区二区久久久| 国产在线高清精品| 亚洲欧洲一区二区三区在线观看| 欧洲午夜精品久久久| 中文字幕视频在线免费欧美日韩综合在线看| 亚洲高清av在线| 国产成人精品亚洲精品| 欧美成人精品一区二区三区| 深夜精品寂寞黄网站在线观看| 日韩中文字幕在线观看| 久久久午夜视频| 欧美在线精品免播放器视频| 色婷婷综合久久久久中文字幕1| 欧美激情视频一区二区三区不卡| 国产精品r级在线| 国产在线日韩在线| 欧美性资源免费| 国产精品成人一区二区三区吃奶| 亚洲一区二区三区成人在线视频精品| 久久手机免费视频| 亚洲深夜福利网站| 国产69久久精品成人| 91精品视频免费看| 国产精品香蕉在线观看| 欧美老女人在线视频| 91精品国产综合久久久久久蜜臀| 久久久精品中文字幕| 国产成+人+综合+亚洲欧洲| 懂色av中文一区二区三区天美| 国产性色av一区二区| 国产91ⅴ在线精品免费观看| 日韩在线视频免费观看高清中文| 国产精品扒开腿做爽爽爽视频| 欧美激情精品久久久久久| 国产精品久久精品| 亚洲欧美日韩国产成人| 欧美视频免费在线观看| 国产欧美精品va在线观看| 国产成人一区二区在线| 中文字幕在线看视频国产欧美| 欧美性猛交xxxx乱大交极品| 日韩精品视频中文在线观看| 国产精品成人观看视频国产奇米| 久精品免费视频| 亚洲国产精品成人精品| 亚洲国产精久久久久久久| 欧美与黑人午夜性猛交久久久| 91精品久久久久| 国产婷婷成人久久av免费高清| 亚洲午夜国产成人av电影男同| 国产亚洲人成网站在线观看| 日韩视频在线免费观看| 91国产精品视频在线| 国内成人精品视频| 亚洲视频电影图片偷拍一区| 亚洲韩国欧洲国产日产av| 中文字幕自拍vr一区二区三区| 8090理伦午夜在线电影| 久久夜色精品亚洲噜噜国产mv| 精品美女久久久久久免费| 亚洲电影免费在线观看| 少妇高潮久久久久久潘金莲| 欧美精品在线免费| www国产91| 亚洲大胆人体av| 国产日韩欧美在线播放| 成人伊人精品色xxxx视频| 最新国产精品拍自在线播放| 久久影视电视剧凤归四时歌| 国产一区玩具在线观看| 精品久久久久久久久久国产| 国产成人精品免高潮费视频| 超薄丝袜一区二区| 丝袜一区二区三区| 国产精品88a∨| 高清视频欧美一级| 国产精品久久久久久久久久99| 在线观看精品国产视频| 欧美国产亚洲精品久久久8v| 国产精自产拍久久久久久| 81精品国产乱码久久久久久| 亚洲91精品在线| 国产欧美精品一区二区三区介绍| 日本不卡免费高清视频| 亚洲精品成人免费| 精品久久中文字幕久久av| 亚洲黄色在线看| 国产午夜精品美女视频明星a级| 欧美成人午夜剧场免费观看|