欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

go實現(xiàn)thrift的網(wǎng)絡(luò)傳輸性能及需要注意問題示例解析

 更新時間:2023年09月06日 14:11:24   作者:donnie4w  
這篇文章主要為大家介紹了go實現(xiàn)thrift的網(wǎng)絡(luò)傳輸性能及需要注意問題示例解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪

thrift簡介

thrift應該是目前支持編程語言種類最多的跨語言 rpc服務(wù)框架, http://thrift.apache.org/

thrift實現(xiàn)了完整的網(wǎng)絡(luò)服務(wù),所以一般使用thrift時,會使用到thrift的服務(wù)框架。當然,也可以用自己已經(jīng)實現(xiàn)的網(wǎng)絡(luò)服務(wù),用io流對接thrift接口的輸入輸出流實現(xiàn)thrift的接入。

無論是用thrift的網(wǎng)絡(luò)實現(xiàn),還是自己實現(xiàn)的網(wǎng)絡(luò)服務(wù),只要對接thrift,在調(diào)用thrift接口實現(xiàn)rpc時,都是走thrift的網(wǎng)絡(luò)傳輸方式。

thrift的網(wǎng)絡(luò)傳輸實現(xiàn)方式 不適合也不支持壓力較大的網(wǎng)絡(luò)傳輸需求。實際上,調(diào)用一次thrift接口,并不是只調(diào)一次網(wǎng)絡(luò)io寫數(shù)據(jù),而是拆分為多次寫數(shù)據(jù)傳送。

調(diào)用一個thrift 的接口發(fā)送數(shù)據(jù)時,thrift會將這個操作拆分為幾個操作:

調(diào)用thrift的方法時:thrift會找到這個方法所在的對象,調(diào)用write方法,

在write方法在,分別對各個參數(shù),依次調(diào)用 writeFieldBeginwriteXXX(具體參數(shù)類型) ,WriteFieldStop 等函數(shù)
每次調(diào)用也同時調(diào)用網(wǎng)絡(luò)io寫相應數(shù)據(jù).

以目前最新的thrift-0.18.1實現(xiàn)為例

go的實現(xiàn)

func (p *ItnetPonMergeArgs) Write(ctx context.Context, oprot thrift.TProtocol) error {
  if err := oprot.WriteStructBegin(ctx, "PonMerge_args"); err != nil {
    return thrift.PrependError(fmt.Sprintf("%T write struct begin error: ", p), err) }
  if p != nil {
    if err := p.writeField1(ctx, oprot); err != nil { return err }
    if err := p.writeField2(ctx, oprot); err != nil { return err }
  }
  if err := oprot.WriteFieldStop(ctx); err != nil {
    return thrift.PrependError("write field stop error: ", err) }
  if err := oprot.WriteStructEnd(ctx); err != nil {
    return thrift.PrependError("write struct stop error: ", err) }
  return nil
}
//第一個參數(shù)
func (p *ItnetPonMergeArgs) writeField1(ctx context.Context, oprot thrift.TProtocol) (err error) {
  if err := oprot.WriteFieldBegin(ctx, "pblist", thrift.LIST, 1); err != nil { //WriteFieldBegin
    return thrift.PrependError(fmt.Sprintf("%T write field begin error 1:pblist: ", p), err) }
  if err := oprot.WriteListBegin(ctx, thrift.STRUCT, len(p.Pblist)); err != nil {
    return thrift.PrependError("error writing list begin: ", err)
  }
  for _, v := range p.Pblist { 
    if err := v.Write(ctx, oprot); err != nil {
      return thrift.PrependError(fmt.Sprintf("%T error writing struct: ", v), err)
    }
  }
  if err := oprot.WriteListEnd(ctx); err != nil {
    return thrift.PrependError("error writing list end: ", err)
  }
  if err := oprot.WriteFieldEnd(ctx); err != nil {
    return thrift.PrependError(fmt.Sprintf("%T write field end error 1:pblist: ", p), err) }
  return err
}
//第二個參數(shù),操作類似第一個參數(shù)
func (p *ItnetPonMergeArgs) writeField2(ctx context.Context, oprot thrift.TProtocol) (err error) {
  if err := oprot.WriteFieldBegin(ctx, "id", thrift.I64, 2); err != nil {
    return thrift.PrependError(fmt.Sprintf("%T write field begin error 2:id: ", p), err) }
  if err := oprot.WriteI64(ctx, int64(p.ID)); err != nil {
  return thrift.PrependError(fmt.Sprintf("%T.id (2) field write error: ", p), err) }
  if err := oprot.WriteFieldEnd(ctx); err != nil {
    return thrift.PrependError(fmt.Sprintf("%T write field end error 2:id: ", p), err) }
  return err
}

調(diào)用Pon(ItnetPonMergeArgs)方法的thrift傳輸順序

Write->

  • writeField1->WriteFieldBegin-> WriteByte-> io
  • -> Write16 -> io
  • ->WriteBinary-> Write32 -> io
  • ->Write -> io
  • writeField2->WriteFieldBegin-> WriteByte -> io
  • ->Write16 -> io
  • ->Write64-> Write -> io
  • WriteFieldStop ->io

可以看到,一次簡單的方法調(diào)用,如果方法中有兩個參數(shù), 則至少有8次io流寫數(shù)據(jù)調(diào)用。

如果參數(shù)多時,或是參數(shù)中一個結(jié)構(gòu)體的變量多時,則會有更多的io流寫數(shù)據(jù)調(diào)用。

在海量的網(wǎng)絡(luò)傳輸中,這樣的傳輸方式,網(wǎng)絡(luò)io流寫數(shù)據(jù)調(diào)用成倍增加,海量網(wǎng)絡(luò)io數(shù)據(jù)寫入導致性能急劇下降。

thrift設(shè)計的傳輸層提供了zlib協(xié)議壓縮,在zlib壓縮發(fā)送的情況下,將數(shù)據(jù)進行了整體壓縮收發(fā),zlib分為2次發(fā)送后,接收端再解壓;

以go為例子:

可以在 compress/flate 看到zlib的寫數(shù)據(jù)最終io寫入調(diào)用:

func (d *compressor) syncFlush() error {
    if d.err != nil {
        return d.err
    }
    d.sync = true
    d.step(d)
    if d.err == nil {
        d.w.writeStoredHeader(0, false)   //第一次調(diào)用
        d.w.flush()                       //第二次調(diào)用
        d.err = d.w.err
    }
    d.sync = false
    return d.err
}
//兩次io數(shù)據(jù)寫入

所以,在海量調(diào)用thrift方法的情況下,zlib模式的性能要遠超非zlib的情況。但是zlib壓縮會比較消耗內(nèi)存,大量使用時可能導致頻繁gc,也可能導致性能下降。當然,即使如此,大部分情況下zlib傳輸依然比非zlib傳輸?shù)男阅芤迷S多。

其他語言的實現(xiàn)

比如 java:

public void write(org.apache.thrift.protocol.TProtocol oprot, SelectByIdxLimit_args struct) throws org.apache.thrift.TException {
        struct.validate();
        oprot.writeStructBegin(STRUCT_DESC);
        if (struct.name != null) {
          oprot.writeFieldBegin(NAME_FIELD_DESC);  //io調(diào)用
          oprot.writeString(struct.name);          //io調(diào)用
          oprot.writeFieldEnd();
        }
        if (struct.column != null) {
          oprot.writeFieldBegin(COLUMN_FIELD_DESC);  //io調(diào)用
          oprot.writeString(struct.column);           //io調(diào)用
          oprot.writeFieldEnd();
        }
        if (struct.value != null) {
          oprot.writeFieldBegin(VALUE_FIELD_DESC);   //io調(diào)用
          {
            oprot.writeListBegin(new org.apache.thrift.protocol.TList(org.apache.thrift.protocol.TType.STRING, struct.value.size()));
            for (java.nio.ByteBuffer _iter49 : struct.value)
            {
              oprot.writeBinary(_iter49);          //io調(diào)用
            }
            oprot.writeListEnd();
          }
          oprot.writeFieldEnd();
        }
        oprot.writeFieldBegin(START_ID_FIELD_DESC); //io調(diào)用
        oprot.writeI64(struct.startId);             //io調(diào)用       
        oprot.writeFieldEnd();
        oprot.writeFieldBegin(LIMIT_FIELD_DESC);    //io調(diào)用
        oprot.writeI64(struct.limit);                //io調(diào)用
        oprot.writeFieldEnd();
        oprot.writeFieldStop();                    //io調(diào)用
        oprot.writeStructEnd();
      }

傳輸方式都是相似的

實現(xiàn)方式各個語言都相似,當然,數(shù)據(jù)寫入順序肯定是一樣的。

以上就是go實現(xiàn)thrift的網(wǎng)絡(luò)及傳輸性能需要注意問題示例解析的詳細內(nèi)容,更多關(guān)于go thrift的網(wǎng)絡(luò)傳輸?shù)馁Y料請關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • GO語言實現(xiàn)TCP服務(wù)器的示例代碼

    GO語言實現(xiàn)TCP服務(wù)器的示例代碼

    這篇文章主要為大家詳細介紹了如何通過GO語言實現(xiàn)TCP服務(wù)器,文中的示例代碼講解詳細,對我們深入了解Go語言有一定的幫助,需要的可以參考一下
    2023-03-03
  • Golang中panic的異常處理

    Golang中panic的異常處理

    本文主要介紹了Golang中panic的異常處理,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2022-08-08
  • Go實踐反向代理ReverseProxy解析

    Go實踐反向代理ReverseProxy解析

    這篇文章主要為大家介紹了Go實踐反向代理示例ReverseProxy解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2023-04-04
  • Go中調(diào)用JS代碼(otto)的實現(xiàn)示例

    Go中調(diào)用JS代碼(otto)的實現(xiàn)示例

    Otto是一個用Go語言實現(xiàn)的JavaScript解釋器,可用于執(zhí)行和操作JavaScript代碼,適合在Go項目中執(zhí)行簡單的JS腳本,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2024-10-10
  • Go中strings的常用方法詳解

    Go中strings的常用方法詳解

    這篇文章主要介紹了Go中strings的常用方法詳解,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2020-03-03
  • Go語言中調(diào)用外部命令的方法總結(jié)

    Go語言中調(diào)用外部命令的方法總結(jié)

    在工作中,我們時不時地會需要在Go中調(diào)用外部命令。本文為大家總結(jié)了Go語言中調(diào)用外部命令的幾種姿勢,感興趣的小伙伴可以跟隨小編一起學習一下
    2022-11-11
  • Golang爬蟲框架 colly的使用

    Golang爬蟲框架 colly的使用

    本文主要介紹了Golang爬蟲框架 colly的使用,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2022-07-07
  • Golang?errgroup?設(shè)計及實現(xiàn)原理解析

    Golang?errgroup?設(shè)計及實現(xiàn)原理解析

    這篇文章主要為大家介紹了Golang?errgroup?設(shè)計及實現(xiàn)原理解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2022-08-08
  • 重學Go語言之如何使用Context

    重學Go語言之如何使用Context

    Context,中文也叫做上下文,Go語言在1.7版本中新增的context包中定義了Context,下面我們就來一起看看如何在Go語言中使用Context吧
    2023-07-07
  • 一文帶你輕松學會Go語言動態(tài)調(diào)用函數(shù)

    一文帶你輕松學會Go語言動態(tài)調(diào)用函數(shù)

    這篇文章主要是帶大家學習一下Go語言是如何動態(tài)調(diào)用函數(shù)的,文中的示例代碼講解詳細,對我們學習Go語言有一定的幫助,需要的可以參考下
    2022-11-11

最新評論