• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            戰(zhàn)魂小筑

            討論群:309800774 知乎關(guān)注:http://zhihu.com/people/sunicdavy 開源項(xiàng)目:https://github.com/davyxu

               :: 首頁 :: 新隨筆 :: 聯(lián)系 :: 聚合  :: 管理 ::
              257 隨筆 :: 0 文章 :: 506 評(píng)論 :: 0 Trackbacks

            準(zhǔn)備:

            http://code.google.com/p/protobuf/下載protobuf-2.5版本

            預(yù)備知識(shí): 已經(jīng)使用過protobuf, 熟練應(yīng)用protobuf序列化在各語言間交互信息

            目標(biāo): 獲取proto內(nèi)容而無需手動(dòng)解析proto文件

            為proto文件添加更多的meta信息, 并在運(yùn)行期獲取.

             

            protoc編譯器準(zhǔn)備

            通過protobuf-2.5的源碼或者從官網(wǎng)下載, 可以獲得protoc的protobuf編譯器, 這個(gè)編譯器由C++編寫, 官方支持完整的protobuf特性. 編譯器默認(rèn)支持C++, python和java 三種語言的代碼生成. 如需生成更多的語言, 可以通過官網(wǎng)的第三方頁面獲取.

             

            protoc插件原理

            但我們?cè)谌粘J褂弥? 可能需要提取proto信息, 例如: 所有的枚舉,消息等信息, 字段名稱和導(dǎo)出號(hào). 自己編寫詞法解析器來做是費(fèi)力不討好的. 官方推薦的方法是使用protoc外掛插件來實(shí)現(xiàn).

            protoc的插件設(shè)計(jì)比較獨(dú)特, 不使用動(dòng)態(tài)鏈接庫或者java的jar包導(dǎo)入方式, 而是直接使用了命令行來交換數(shù)據(jù).查看protobuf源碼我們可以發(fā)現(xiàn)這樣一個(gè)文件:

            protobuf-2.5.0\src\google\protobuf\descriptor.proto

            這個(gè)文件描述了一個(gè)proto文件的格式, 消息組成及枚舉等完整信息. 這是一種自我描述的方法.

            在找到這樣一個(gè)文件

            protobuf-2.5.0\src\google\protobuf\compiler\plugin.proto

            這樣一個(gè)文件描述: 插件如何與protoc進(jìn)行交互的協(xié)議

            protoc編譯器在給定指定proto文件及搜索路徑后, 將各種信息填充為descriptor.proto描述的結(jié)構(gòu)后通過CodeGeneratorRequest消息系列化為二進(jìn)制流后輸出到命令行. 插件只用捕獲protoc命令行輸出的二進(jìn)制流, 序列化化回CodeGeneratorRequest即可獲得解析后的proto文件內(nèi)容

            這里需要注意的是: 插件可執(zhí)行文件很有講究, 必須為protoc-gen-$NAME,  而且輸出文件名參數(shù)必須為--${NAME}_out

            看一個(gè)栗子:

            protoc.exe foo.proto --plugin=protoc-gen-go=..\tools\protoc-gen-go.exe --go_out foo.go --proto_path "."

            這個(gè)栗子里: $NAME=go

            protoc將foo.proto文件(搜索路徑為當(dāng)前路徑)的內(nèi)容通過命令行輸出給位于..\tools\的插件protoc-gen-go.exe,  輸出文件名字為 foo.go

            descriptor.proto信息挖掘

            我們注意到在descriptor.proto文件中包含有這樣的一個(gè)message: SourceCodeInfo, 這個(gè)消息體里有如下字段

            optional string leading_comments = 3;
                optional string trailing_comments = 4;

            這兩個(gè)字段對(duì)于我們獲取proto文件的meta信息尤為重要, 所謂的meta信息, 理解理解為C#語言中的attribute

            這個(gè)attribute功能可以為一個(gè)字段, 一個(gè)消息擴(kuò)充一些描述. 比如: 當(dāng)一個(gè)字段通過反射顯示在gui上時(shí), gui需要獲取這個(gè)字段的中文描述

            那么只需要如下編寫

            optional int32 somevalue = 1 //@ desc=”中文描述”

            位于字段尾部的描述, 會(huì)被填充到SourceCodeInfo的 trailing_comments中, 而位于字段上方的字段, 會(huì)被填充到leading_comments中

             

            SourceCodeInfo 并沒有直接掛載在message或者字段的附近, 而是通過其下的path字段來描述與字段的關(guān)系, 這是個(gè)極為麻煩的設(shè)計(jì).

            其原理如下:

            假設(shè)我有如下一個(gè)message

            message foo

            {

                 optional int32 v = 1;  // comments

            }

            要獲取v后的注釋, 對(duì)應(yīng)的path為 4, 0, 2, 0

            4 表示descriptor中message_type所在的序號(hào),由于message_type對(duì)應(yīng)的類型DescriptorProto是一個(gè)數(shù)組, 所以0表示foo是在FileDescriptorProto的message_type數(shù)組類型的索引為0;

            如此類推: 2, 0 表示 v在DescriptorProto結(jié)構(gòu)體的field成員序號(hào)為2的數(shù)組元素的索引為0

             

            如果需要更多的參考, 可以獲取https://github.com/golang/protobuf

            github.com\golang\protobuf\protoc-gen-go工程內(nèi)有詳細(xì)代碼解析

            posted on 2015-03-01 13:49 戰(zhàn)魂小筑 閱讀(11078) 評(píng)論(0)  編輯 收藏 引用 所屬分類: 腳本技術(shù)工具使用及設(shè)計(jì)
            久久久久无码专区亚洲av| 久久久久人妻一区精品色| 久久精品99无色码中文字幕| 久久亚洲色一区二区三区| 中文字幕无码久久人妻| 亚洲精品无码久久千人斩| 99热精品久久只有精品| 国产成人久久精品一区二区三区| 久久99国产精品久久99| 97久久国产综合精品女不卡| 国产亚洲精久久久久久无码AV| 久久久亚洲AV波多野结衣| 99久久精品免费看国产免费| 亚洲国产精品无码久久一区二区| 久久久久亚洲AV综合波多野结衣 | 久久久久久无码国产精品中文字幕| 理论片午午伦夜理片久久 | 久久人人爽人人爽人人片AV东京热| 99久久er这里只有精品18| 国内精品伊人久久久久| 亚洲欧美成人综合久久久| 久久久久国产精品三级网| 国产精品视频久久久| 日韩久久久久久中文人妻| 区久久AAA片69亚洲| 久久天天躁狠狠躁夜夜不卡| 久久精品视频网| 99精品国产在热久久| 久久青青草原亚洲av无码app | 久久精品人人做人人妻人人玩| 蜜桃麻豆WWW久久囤产精品| 久久人人爽人爽人人爽av| 久久久艹| 亚洲一级Av无码毛片久久精品| 久久精品夜色噜噜亚洲A∨| 国产亚洲精午夜久久久久久| 99久久精品国产一区二区| 国产精品99久久久久久董美香 | 国产精品一区二区久久精品| 国产精品99久久免费观看| 国产精品久久久久久搜索|