最近的項目中,一直使用Json作數據傳輸。Json用起來的確很方便。但相對於protobuf數據量更大些。作一個移動端應用,爲用戶省點流量仍是頗有必要的。正好也能夠學習一下protobuf的使用linux
跟Json相比protobuf性能更高,更加規範git
編解碼速度快,數據體積小github
使用統一的規範,不用再擔憂大小寫不一樣致使解析失敗等蛋疼的問題了golang
但也失去了一些便利性shell
改動協議字段,須要從新生成文件。windows
數據沒有可讀性服務器
在go中使用protobuf,有兩個可選用的包goprotobuf(go官方出品)和gogoprotobuf。
gogoprotobuf徹底兼容google protobuf,它生成的代碼質量和編解碼性能均比goprotobuf高一些tcp
首先去https://github.com/google/pro... 上下載protobuf的編譯器protoc,windows上能夠直接下到exe文件(linux則須要編譯),最後將下載好的可執行文件拷貝到$GOPATH的bin目錄下($GOPATH/bin目錄最好添加到系統環境變量裏)性能
go get github.com/golang/protobuf/proto
go get github.com/golang/protobuf/protoc-gen-go
protoc --go_out=. *.proto
gogoprotobuf有兩個插件可使用學習
protoc-gen-gogo:和protoc-gen-go生成的文件差很少,性能也幾乎同樣(稍微快一點點)
protoc-gen-gofast:生成的文件更復雜,性能也更高(快5-7倍)
//gogo go get github.com/gogo/protobuf/protoc-gen-gogo //gofast go get github.com/gogo/protobuf/protoc-gen-gofast
go get github.com/gogo/protobuf/proto go get github.com/gogo/protobuf/gogoproto //這個不裝也不要緊
//gogo protoc --gogo_out=. *.proto //gofast protoc --gofast_out=. *.proto
這裏只是簡單的用go test測試了一下
//goprotobuf "編碼":447ns/op "解碼":422ns/op //gogoprotobuf-go "編碼":433ns/op "解碼":427ns/op //gogoprotobuf-fast "編碼":112ns/op "解碼":112ns/op
syntax = "proto3"; //指定版本,必需要寫(proto三、proto2) package proto; enum FOO { X = 0; }; //message是固定的。UserInfo是類名,能夠隨意指定,符合規範便可 message UserInfo{ string message = 1; //消息 int32 length = 2; //消息大小 int32 cnt = 3; //消息計數 }
package main import ( "bufio" "fmt" "net" "os" stProto "proto" "time" //protobuf編解碼庫,下面兩個庫是相互兼容的,可使用其中任意一個 "github.com/golang/protobuf/proto" //"github.com/gogo/protobuf/proto" ) func main() { strIP := "localhost:6600" var conn net.Conn var err error //鏈接服務器 for conn, err = net.Dial("tcp", strIP); err != nil; conn, err = net.Dial("tcp", strIP) { fmt.Println("connect", strIP, "fail") time.Sleep(time.Second) fmt.Println("reconnect...") } fmt.Println("connect", strIP, "success") defer conn.Close() //發送消息 cnt := 0 sender := bufio.NewScanner(os.Stdin) for sender.Scan() { cnt++ stSend := &stProto.UserInfo{ Message: sender.Text(), Length: *proto.Int(len(sender.Text())), Cnt: *proto.Int(cnt), } //protobuf編碼 pData, err := proto.Marshal(stSend) if err != nil { panic(err) } //發送 conn.Write(pData) if sender.Text() == "stop" { return } } }
package main import ( "fmt" "net" "os" stProto "proto" //protobuf編解碼庫,下面兩個庫是相互兼容的,可使用其中任意一個 "github.com/golang/protobuf/proto" //"github.com/gogo/protobuf/proto" ) func main() { //監聽 listener, err := net.Listen("tcp", "localhost:6600") if err != nil { panic(err) } for { conn, err := listener.Accept() if err != nil { panic(err) } fmt.Println("new connect", conn.RemoteAddr()) go readMessage(conn) } } //接收消息 func readMessage(conn net.Conn) { defer conn.Close() buf := make([]byte, 4096, 4096) for { //讀消息 cnt, err := conn.Read(buf) if err != nil { panic(err) } stReceive := &stProto.UserInfo{} pData := buf[:cnt] //protobuf解碼 err = proto.Unmarshal(pData, stReceive) if err != nil { panic(err) } fmt.Println("receive", conn.RemoteAddr(), stReceive) if stReceive.Message == "stop" { os.Exit(1) } } }