如下是照着python操做protobuf進行的protobuf-python的學習筆記:java
首先是protobuf的下載與安裝:python
1 因爲google被牆,因此去github上面搜索了一下protobuf,因而找到了protobuf的git頁:protobuf on github。git
2 能夠本身將protobuf下載下來進行編譯,也能夠直接用人家編譯好的發行版,這個要看我的,我選擇的是用編譯好的發行版,下載地址在這裏:protobuf realse version。github
記得要同時下載兩個東西,例如我下載的是:protobuf-2.6.1.zip 和 protobuf-2.6.1-win32.zip。swift
前者包含了protobuf與各語言(java,python)之間的protobuf運行時庫,這個在轉換的時候須要用到,至關與protobuf與各語言之間的協定格式。ide
後者包含了一個文件 protoc.exe,編譯工具來的。工具
3 因爲咱們是要學習protobuf-python來着,因此解壓玩protobuf-2.6.1.zip以後,咱們能夠在文件夾下面找到一個python文件夾,在其中會看到一個setup.py。學習
咱們依次運行:python setup.py build 和 python setup.py installui
有沒有成功在控制檯能夠看獲得。this
這樣環境就安裝好了(python 建議選擇2.7版本)。
接着就是編寫和轉換proto文件:
1 新建一個addressbook.proto文件,這個文件徹底是按照原帖編寫的,以下:
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 Person person = 1; }
2 利用protoc.exe編譯proto文件,以下:
其中 -I是源文件目錄 --python_out是文件輸出目錄 最後的參數addressbook.proto是你須要編譯的協議文件,
編譯好以後你就會在目標目錄裏面看到輸出的結果文件,以下:
1 # Generated by the protocol buffer compiler. DO NOT EDIT! 2 # source: addressbook.proto 3 4 import sys 5 _b=sys.version_info[0]<3 and (lambda x:x) or (lambda x:x.encode('latin1')) 6 from google.protobuf import descriptor as _descriptor 7 from google.protobuf import message as _message 8 from google.protobuf import reflection as _reflection 9 from google.protobuf import symbol_database as _symbol_database 10 from google.protobuf import descriptor_pb2 11 # @@protoc_insertion_point(imports) 12 13 _sym_db = _symbol_database.Default() 14 15 16 17 18 DESCRIPTOR = _descriptor.FileDescriptor( 19 name='addressbook.proto', 20 package='tutorial', 21 serialized_pb=_b('\n\x11\x61\x64\x64ressbook.proto\x12\x08tutorial\"\xda\x01\n\x06Person\x12\x0c\n\x04name\x18\x01 \x02(\t\x12\n\n\x02id\x18\x02 \x02(\x05\x12\r\n\x05\x65mail\x18\x03 \x01(\t\x12+\n\x05phone\x18\x04 \x03(\x0b\x32\x1c.tutorial.Person.PhoneNumber\x1aM\n\x0bPhoneNumber\x12\x0e\n\x06number\x18\x01 \x02(\t\x12.\n\x04type\x18\x02 \x01(\x0e\x32\x1a.tutorial.Person.PhoneType:\x04HOME\"+\n\tPhoneType\x12\n\n\x06MOBILE\x10\x00\x12\x08\n\x04HOME\x10\x01\x12\x08\n\x04WORK\x10\x02\"/\n\x0b\x41\x64\x64ressBook\x12 \n\x06person\x18\x01 \x03(\x0b\x32\x10.tutorial.Person') 22 ) 23 _sym_db.RegisterFileDescriptor(DESCRIPTOR) 24 25 26 27 _PERSON_PHONETYPE = _descriptor.EnumDescriptor( 28 name='PhoneType', 29 full_name='tutorial.Person.PhoneType', 30 filename=None, 31 file=DESCRIPTOR, 32 values=[ 33 _descriptor.EnumValueDescriptor( 34 name='MOBILE', index=0, number=0, 35 options=None, 36 type=None), 37 _descriptor.EnumValueDescriptor( 38 name='HOME', index=1, number=1, 39 options=None, 40 type=None), 41 _descriptor.EnumValueDescriptor( 42 name='WORK', index=2, number=2, 43 options=None, 44 type=None), 45 ], 46 containing_type=None, 47 options=None, 48 serialized_start=207, 49 serialized_end=250, 50 ) 51 _sym_db.RegisterEnumDescriptor(_PERSON_PHONETYPE) 52 53 54 _PERSON_PHONENUMBER = _descriptor.Descriptor( 55 name='PhoneNumber', 56 full_name='tutorial.Person.PhoneNumber', 57 filename=None, 58 file=DESCRIPTOR, 59 containing_type=None, 60 fields=[ 61 _descriptor.FieldDescriptor( 62 name='number', full_name='tutorial.Person.PhoneNumber.number', index=0, 63 number=1, type=9, cpp_type=9, label=2, 64 has_default_value=False, default_value=_b("").decode('utf-8'), 65 message_type=None, enum_type=None, containing_type=None, 66 is_extension=False, extension_scope=None, 67 options=None), 68 _descriptor.FieldDescriptor( 69 name='type', full_name='tutorial.Person.PhoneNumber.type', index=1, 70 number=2, type=14, cpp_type=8, label=1, 71 has_default_value=True, default_value=1, 72 message_type=None, enum_type=None, containing_type=None, 73 is_extension=False, extension_scope=None, 74 options=None), 75 ], 76 extensions=[ 77 ], 78 nested_types=[], 79 enum_types=[ 80 ], 81 options=None, 82 is_extendable=False, 83 extension_ranges=[], 84 oneofs=[ 85 ], 86 serialized_start=128, 87 serialized_end=205, 88 ) 89 90 _PERSON = _descriptor.Descriptor( 91 name='Person', 92 full_name='tutorial.Person', 93 filename=None, 94 file=DESCRIPTOR, 95 containing_type=None, 96 fields=[ 97 _descriptor.FieldDescriptor( 98 name='name', full_name='tutorial.Person.name', index=0, 99 number=1, type=9, cpp_type=9, label=2, 100 has_default_value=False, default_value=_b("").decode('utf-8'), 101 message_type=None, enum_type=None, containing_type=None, 102 is_extension=False, extension_scope=None, 103 options=None), 104 _descriptor.FieldDescriptor( 105 name='id', full_name='tutorial.Person.id', index=1, 106 number=2, type=5, cpp_type=1, label=2, 107 has_default_value=False, default_value=0, 108 message_type=None, enum_type=None, containing_type=None, 109 is_extension=False, extension_scope=None, 110 options=None), 111 _descriptor.FieldDescriptor( 112 name='email', full_name='tutorial.Person.email', index=2, 113 number=3, type=9, cpp_type=9, label=1, 114 has_default_value=False, default_value=_b("").decode('utf-8'), 115 message_type=None, enum_type=None, containing_type=None, 116 is_extension=False, extension_scope=None, 117 options=None), 118 _descriptor.FieldDescriptor( 119 name='phone', full_name='tutorial.Person.phone', index=3, 120 number=4, type=11, cpp_type=10, label=3, 121 has_default_value=False, default_value=[], 122 message_type=None, enum_type=None, containing_type=None, 123 is_extension=False, extension_scope=None, 124 options=None), 125 ], 126 extensions=[ 127 ], 128 nested_types=[_PERSON_PHONENUMBER, ], 129 enum_types=[ 130 _PERSON_PHONETYPE, 131 ], 132 options=None, 133 is_extendable=False, 134 extension_ranges=[], 135 oneofs=[ 136 ], 137 serialized_start=32, 138 serialized_end=250, 139 ) 140 141 142 _ADDRESSBOOK = _descriptor.Descriptor( 143 name='AddressBook', 144 full_name='tutorial.AddressBook', 145 filename=None, 146 file=DESCRIPTOR, 147 containing_type=None, 148 fields=[ 149 _descriptor.FieldDescriptor( 150 name='person', full_name='tutorial.AddressBook.person', index=0, 151 number=1, type=11, cpp_type=10, label=3, 152 has_default_value=False, default_value=[], 153 message_type=None, enum_type=None, containing_type=None, 154 is_extension=False, extension_scope=None, 155 options=None), 156 ], 157 extensions=[ 158 ], 159 nested_types=[], 160 enum_types=[ 161 ], 162 options=None, 163 is_extendable=False, 164 extension_ranges=[], 165 oneofs=[ 166 ], 167 serialized_start=252, 168 serialized_end=299, 169 ) 170 171 _PERSON_PHONENUMBER.fields_by_name['type'].enum_type = _PERSON_PHONETYPE 172 _PERSON_PHONENUMBER.containing_type = _PERSON 173 _PERSON.fields_by_name['phone'].message_type = _PERSON_PHONENUMBER 174 _PERSON_PHONETYPE.containing_type = _PERSON 175 _ADDRESSBOOK.fields_by_name['person'].message_type = _PERSON 176 DESCRIPTOR.message_types_by_name['Person'] = _PERSON 177 DESCRIPTOR.message_types_by_name['AddressBook'] = _ADDRESSBOOK 178 179 Person = _reflection.GeneratedProtocolMessageType('Person', (_message.Message,), dict( 180 181 PhoneNumber = _reflection.GeneratedProtocolMessageType('PhoneNumber', (_message.Message,), dict( 182 DESCRIPTOR = _PERSON_PHONENUMBER, 183 __module__ = 'addressbook_pb2' 184 # @@protoc_insertion_point(class_scope:tutorial.Person.PhoneNumber) 185 )) 186 , 187 DESCRIPTOR = _PERSON, 188 __module__ = 'addressbook_pb2' 189 # @@protoc_insertion_point(class_scope:tutorial.Person) 190 )) 191 _sym_db.RegisterMessage(Person) 192 _sym_db.RegisterMessage(Person.PhoneNumber) 193 194 AddressBook = _reflection.GeneratedProtocolMessageType('AddressBook', (_message.Message,), dict( 195 DESCRIPTOR = _ADDRESSBOOK, 196 __module__ = 'addressbook_pb2' 197 # @@protoc_insertion_point(class_scope:tutorial.AddressBook) 198 )) 199 _sym_db.RegisterMessage(AddressBook) 200 201 202 # @@protoc_insertion_point(module_scope)
而後咱們就可使用生成的python文件了,
1 首先新建一個程序文檔,代碼仍是來自原帖的:
1 import addressbook_pb2 2 import sys 3 4 def PromptForAddress(person): 5 person.id = int(raw_input("Please input the id of this person...")) 6 person.name = raw_input("Please input a name for the person...") 7 email = raw_input("Please enter the email address of the person....") 8 9 if email != "": 10 person.email = email 11 pass 12 13 while True: 14 number = raw_input("Enter a phone number :") 15 if number == "": 16 break 17 pass 18 19 phone_number = person.phone.add() 20 phone_number.number = number 21 22 23 type = raw_input("Is this a mobile, home, or work phone?") 24 if type == "mobile": 25 phone_number.type = addressbook_pb2.Person.MOBILE 26 elif type == "home": 27 phone_number.type = addressbook_pb2.Person.HOME 28 elif type == "work": 29 phone_number.type = addressbook_pb2.Person.WORK 30 else: 31 print("Unknown phont type; leaving as default value.") 32 pass 33 pass 34 pass 35 36 37 if len(sys.argv) != 2: 38 print("Usage:", sys.argv[0], "ADDRESS_BOOK_FILE") 39 sys.exit(-1) 40 pass 41 print("What am i doing....") 42 address_book = addressbook_pb2.AddressBook() 43 44 try: 45 f = open(sys.argv[1], "rb") 46 address_book.ParseFromString(f.read()) 47 f.close() 48 except IOError, e: 49 print(sys.argv[1] + " : File not found. Creating a new file") 50 pass 51 52 PromptForAddress(address_book.person.add()) 53 54 f = open(sys.argv[1], "wb") 55 f.write(address_book.SerializeToString()) 56 f.close()
2 而後運行這個文件,並按照提示進行操做:
而後你就能夠在你的文件夾下面發現了保存的beach文件。