黄视频网站在线免费观看-黄视频网站在线看-黄视频网站在线观看-黄视频网站免费看-黄视频网站免费观看-黄视频网站免费

千鋒教育-做有情懷、有良心、有品質的職業教育機構

手機站
千鋒教育

千鋒學習站 | 隨時隨地免費學

千鋒教育

掃一掃進入千鋒手機站

領取全套視頻
千鋒教育

關注千鋒學習站小程序
隨時隨地免費學習課程

當前位置:首頁  >  技術干貨  > 如何使用golang實現自定義RPC框架

如何使用golang實現自定義RPC框架

來源:千鋒教育
發布人:xqq
時間: 2023-12-27 05:03:46 1703624626

如何使用golang實現自定義RPC框架

RPC (Remote Procedure Call)是一種遠程調用協議,通過網絡傳輸,使得程序能夠像本地調用一樣調用遠程服務。在現代微服務架構中,RPC協議被廣泛使用。golang通過標準庫的net/rpc包提供了一套RPC框架,但是這個框架無法滿足一些特定的業務需求,本文就來介紹如何使用golang自己實現一個RPC框架。

1. 基本概念

在實現自定義RPC框架之前,需要先了解以下幾個基本概念:

- Service:RPC調用的服務,即提供RPC服務的函數集合。

- Method:Service中的方法,即具體的RPC調用方法。

- Codec:序列化和反序列化的方法,將調用的參數和返回值序列化成二進制數據,以便通過網絡傳輸。

- Transport:網絡傳輸協議,用于將序列化后的二進制數據通過網絡傳輸到遠程服務。

2. 實現步驟

接下來我們就來實現一個簡單的自定義RPC框架,步驟如下:

- 定義Service和Method

- 實現Codec

- 實現Transport

- 完成框架

2.1 定義Service和Method

我們以一個簡單的計算器服務為例,在服務端提供兩個方法Add和Multiply,客戶端可以通過RPC調用這兩個方法。

定義服務:

`go

// 定義CalculatorService接口

type CalculatorService interface {

Add(int, int) int

Multiply(int, int) int

}

// 實現具體的CalculatorService

type CalculatorServiceImpl struct {}

func (c *CalculatorServiceImpl) Add(a, b int) int {

return a + b

}

func (c *CalculatorServiceImpl) Multiply(a, b int) int {

return a * b

}

定義Service和Method之后,接下來需要定義一個struct來存儲Service和其對應的Method。同時,定義一個Register方法,用于注冊新的Service和Method。`gotype Server struct {    services map*service}type service struct {    typ    reflect.Type    method map*methodType}type methodType struct {    method    reflect.Method    ArgType   reflect.Type    ReplyType reflect.Type}func (s *Server) Register(receiver interface{}) error {    service := new(service)    service.typ = reflect.TypeOf(receiver).Elem()    service.method = make(map*methodType)    for i := 0; i < service.typ.NumMethod(); i++ {        method := service.typ.Method(i)        mType := method.Type        if mType.NumIn() != 3 || mType.NumOut() != 1 {            continue        }        argType := mType.In(1)        replyType := mType.In(2)        if !isExportedOrBuiltinType(argType) || !isExportedOrBuiltinType(replyType) {            continue        }        service.method = &methodType{            method:    method,            ArgType:   argType,            ReplyType: replyType,        }    }    s.services = service    return nil}func isExportedOrBuiltinType(t reflect.Type) bool {    pkgPath := t.PkgPath()    return pkgPath == "" || pkgPath == "builtin"}

在Register方法中,循環遍歷service.typ中的所有方法,將滿足條件的方法添加到service.method中。最后將service添加到Server.services中。

2.2 實現Codec

Codec用于將調用的參數和返回值序列化成二進制數據,以便通過網絡傳輸。

在這里,我們使用golang的標準庫encoding/gob實現Codec。Gob是golang標準庫中的編解碼庫,支持任意類型的編解碼和傳輸,比JSON和XML更高效。在實現Codec之前,需要先定義一個request結構體和response結構體,用于存儲調用信息和返回信息。

`go

type request struct {

ServiceMethod string // 形如"Service.Method"

Seq uint64 // 請求序列號

Args byte // 客戶端傳遞的參數

}

type response struct {

Seq uint64 // 請求序列號

ServiceMethod string // 形如"Service.Method"

Error string // 存儲錯誤信息

Reply byte // 存儲響應參數

}

接下來實現Codec,具體實現代碼如下:`gotype Codec struct {    conn io.ReadWriteCloser    dec  *gob.Decoder    enc  *gob.Encoder    mutex sync.Mutex    ids   uint64    pending map*call}type call struct {    req  *request    resp *response    done chan *call}func (c *Codec) WriteRequest(method string, args interface{}) (uint64, error) {    c.mutex.Lock()    defer c.mutex.Unlock()    id := c.ids    c.ids++    req := &request{        ServiceMethod: method,        Seq:           id,    }    buf := bytes.NewBuffer(nil)    enc := gob.NewEncoder(buf)    if err := enc.Encode(args); err != nil {        return 0, err    }    req.Args = buf.Bytes()    call := &call{        req:  req,        resp: new(response),        done: make(chan *call),    }    c.pending = call    if err := c.enc.Encode(req); err != nil {        delete(c.pending, id)        return 0, err    }    return id, nil}func (c *Codec) ReadResponseHeader() (*rpc.Response, error) {    c.mutex.Lock()    defer c.mutex.Unlock()    var resp response    if err := c.dec.Decode(&resp); err != nil {        return nil, err    }    call := c.pending    delete(c.pending, resp.Seq)    call.resp = &resp    call.done <- call    return &rpc.Response{        ServiceMethod: resp.ServiceMethod,        Seq:           resp.Seq,        Error:         errors.New(resp.Error),    }, nil}func (c *Codec) ReadResponseBody(x interface{}) error {    c.mutex.Lock()    defer c.mutex.Unlock()    call := <-c.pending.done    if call.resp.Error != "" {        return errors.New(call.resp.Error)    }    dec := gob.NewDecoder(bytes.NewBuffer(call.resp.Reply))    return dec.Decode(x)}

在上面的代碼中,我們使用了一個pending map來存儲請求的序列號和請求的返回值。在WriteRequest方法中,我們將請求信息編碼成二進制數據,然后將請求信息和該請求的channel存儲到pending中。在ReadResponseHeader和ReadResponseBody方法中,我們根據pending中的請求序列號獲取該請求對應的call,然后將call.resp進行解碼后返回。

2.3 實現Transport

Transport用于將序列化后的二進制數據通過網絡傳輸到遠程服務。

在golang中,可以使用net包來實現簡單的Socket編程。在這里,我們通過net.Dial建立連接后,將Codec中序列化后的數據通過Socket發送到遠程服務端。

`go

type Transport struct {

conn io.ReadWriteCloser

}

func (t *Transport) Dial(network, address string) error {

conn, err := net.Dial(network, address)

if err != nil {

return err

}

t.conn = conn

return nil

}

func (t *Transport) Close() error {

return t.conn.Close()

}

func (t *Transport) Codec() rpc.ClientCodec {

return &Codec{

conn: t.conn,

dec: gob.NewDecoder(t.conn),

enc: gob.NewEncoder(t.conn),

pending: make(map*call),

}

}

2.4 完成框架最后,我們完成自定義RPC框架的實現。具體代碼如下:`gotype Server struct {    services map*service}type service struct {    typ    reflect.Type    method map*methodType}type methodType struct {    method    reflect.Method    ArgType   reflect.Type    ReplyType reflect.Type}func (s *Server) Register(receiver interface{}) error {    service := new(service)    service.typ = reflect.TypeOf(receiver).Elem()    service.method = make(map*methodType)    for i := 0; i < service.typ.NumMethod(); i++ {        method := service.typ.Method(i)        mType := method.Type        if mType.NumIn() != 3 || mType.NumOut() != 1 {            continue        }        argType := mType.In(1)        replyType := mType.In(2)        if !isExportedOrBuiltinType(argType) || !isExportedOrBuiltinType(replyType) {            continue        }        service.method = &methodType{            method:    method,            ArgType:   argType,            ReplyType: replyType,        }    }    s.services = service    return nil}func (s *Server) ServeCodec(codec rpc.ServerCodec) error {    for {        req, err := codec.ReadRequestHeader()        if err != nil {            if err != io.EOF && err != io.ErrUnexpectedEOF {                log.Println("rpc server:", err)            }            return err        }        serviceMethod := req.ServiceMethod        dot := strings.LastIndex(serviceMethod, ".")        if dot < 0 {            err := errors.New("rpc server: service/method request ill-formed: " + serviceMethod)            log.Println(err.Error())            resp := &rpc.Response{                ServiceMethod: serviceMethod,                Seq:           req.Seq,                Error:         err.Error(),            }            codec.WriteResponse(resp, nil)            continue        }        serviceName, methodName := serviceMethod, serviceMethod        service, ok := s.services        if !ok {            err := errors.New("rpc server: can't find service " + serviceName)            log.Println(err.Error())            resp := &rpc.Response{                ServiceMethod: serviceMethod,                Seq:           req.Seq,                Error:         err.Error(),            }            codec.WriteResponse(resp, nil)            continue        }        mtype, ok := service.method        if !ok {            err := errors.New("rpc server: can't find method " + methodName)            log.Println(err.Error())            resp := &rpc.Response{                ServiceMethod: serviceMethod,                Seq:           req.Seq,                Error:         err.Error(),            }            codec.WriteResponse(resp, nil)            continue        }        argv := reflect.New(mtype.ArgType)        replyv := reflect.New(mtype.ReplyType).Elem()        if err = codec.ReadRequestBody(argv.Interface()); err != nil {            log.Println("rpc server: ", err)            resp := &rpc.Response{                ServiceMethod: serviceMethod,                Seq:           req.Seq,                Error:         err.Error(),            }            codec.WriteResponse(resp, nil)            continue        }        // Call the service method.        returnValues := mtype.method.Func.Call(reflect.Value{            reflect.ValueOf(service),            reflect.ValueOf(argv.Interface()),            replyv,        })        // The return value for the method is an error.        errInter := returnValues.Interface()        if errInter != nil {            err := errInter.(error)            log.Println("rpc server: ", err)            resp := &rpc.Response{                ServiceMethod: serviceMethod,                Seq:           req.Seq,                Error:         err.Error(),            }            codec.WriteResponse(resp, nil)            continue        }        resp := &rpc.Response{            ServiceMethod: serviceMethod,            Seq:           req.Seq,            Error:         "",        }        if err = codec.WriteResponse(resp, replyv.Interface()); err != nil {            log.Println("rpc server: ", err)        }    }}func (s *Server) ServeTransport(transport *Transport) error {    codec := transport.Codec()    defer transport.Close()    return s.ServeCodec(codec)}func isExportedOrBuiltinType(t reflect.Type) bool {    pkgPath := t.PkgPath()    return pkgPath == "" || pkgPath == "builtin"}

在上面的代碼中,我們定義了一個Server結構體,用于注冊Service和Method,同時實現ServeCodec和ServeTransport方法,用于在服務端處理RPC請求。

3. 測試

完成自定義RPC框架的實現之后,我們需要對其進行測試。下面我們將分別在服務端和客戶端使用該RPC框架。

服務端代碼:

`go

func main() {

server := new(Server)

server.services = make(map*service)

_ = server.Register(&CalculatorServiceImpl{})

transport := new(Transport)

_ = transport.Dial("tcp", "localhost:8080")

defer transport.Close()

log.Fatal(server.ServeTransport(transport))

}

在服務端,我們首先通過Server.Register方法注冊了一個CalculatorServiceImpl服務,然后使用Transport.Dial方法連接到特定的地址。客戶端代碼:`gofunc main() {    transport := new(Transport)    _ = transport.Dial("tcp", "localhost:8080")    defer transport.Close()    client := rpc.NewClientWithCodec(transport.Codec())    var res int    err := client.Call("CalculatorService.Add", int{10, 20}, &res)    if err != nil {        log.Fatal(err)    }    log.Printf("Add(10, 20) = %d", res)    var mul int    err = client.Call("CalculatorService.Multiply", int{10, 20}, &mul)    if err != nil {        log.Fatal(err)    }    log.Printf("Multiply(10, 20) = %d", mul)}

在客戶端,我們首先通過Transport.Dial方法連接到服務端,然后通過rpc.NewClientWithCodec方法創建一個客戶端,并使用client.Call方法調用服務端的方法。

最后,我們啟動服務端和客戶端,可以看到客戶端成功調用了服務端提供的Add和Multiply方法。

4. 總結

本文介紹了如何使用golang實現自定義RPC框架,包括定義Service和Method,實現Codec和Transport,完成框架等步驟,并通過一個簡單的計算器服務對該RPC框架進行了測試。該自定義RPC框架適用于一些特定的業務需求,可以滿足不同的RPC調用場景。

以上就是IT培訓機構千鋒教育提供的相關內容,如果您有web前端培訓鴻蒙開發培訓python培訓linux培訓,java培訓,UI設計培訓等需求,歡迎隨時聯系千鋒教育。

tags:
聲明:本站稿件版權均屬千鋒教育所有,未經許可不得擅自轉載。
10年以上業內強師集結,手把手帶你蛻變精英
請您保持通訊暢通,專屬學習老師24小時內將與您1V1溝通
免費領取
今日已有369人領取成功
劉同學 138****2860 剛剛成功領取
王同學 131****2015 剛剛成功領取
張同學 133****4652 剛剛成功領取
李同學 135****8607 剛剛成功領取
楊同學 132****5667 剛剛成功領取
岳同學 134****6652 剛剛成功領取
梁同學 157****2950 剛剛成功領取
劉同學 189****1015 剛剛成功領取
張同學 155****4678 剛剛成功領取
鄒同學 139****2907 剛剛成功領取
董同學 138****2867 剛剛成功領取
周同學 136****3602 剛剛成功領取
相關推薦HOT
青青青草影院| 国产a免费观看| 国产91精品一区| 国产视频久久久| 成人影视在线播放| 免费的黄色小视频| 999久久66久6只有精品| 国产一区二区精品久久| 黄视频网站在线看| 国产亚洲精品成人a在线| 成人免费观看的视频黄页| 精品国产一区二区三区精东影业| 九九九在线视频| 欧美爱爱动态| 九九久久国产精品| 久久国产影院| 国产伦理精品| 国产成人精品影视| a级黄色毛片免费播放视频| 日韩中文字幕一区二区不卡| 香蕉视频久久| 色综合久久天天综合观看| 亚洲第一视频在线播放| 成人a大片在线观看| 日韩专区一区| 精品视频一区二区| 欧美a级成人淫片免费看| 国产麻豆精品免费密入口| 日本免费乱人伦在线观看| 欧美另类videosbestsex久久| 天堂网中文在线| 国产a免费观看| 日韩一级黄色大片| 高清一级片| 国产不卡在线观看| 麻豆系列 在线视频| 国产欧美精品午夜在线播放| 日韩欧美一二三区| 天天做日日爱夜夜爽| 日韩中文字幕在线播放| a级精品九九九大片免费看| 色综合久久天天综线观看| 精品视频在线看 | 香蕉视频亚洲一级| 91麻豆精品国产自产在线观看一区 | 成人影院一区二区三区| 成人a级高清视频在线观看| 久久国产精品只做精品| 亚洲精品影院久久久久久| 国产伦精品一区二区三区无广告 | 九九九在线视频| 国产不卡精品一区二区三区| 国产高清视频免费观看| 日本伦理网站| 亚欧成人乱码一区二区| 久久99这里只有精品国产| 国产a毛片| 韩国毛片| 久久福利影视| 亚洲第一色在线| 欧美一区二区三区在线观看| 台湾美女古装一级毛片| 中文字幕97| 亚洲精品影院| 可以在线看黄的网站| 色综合久久天天综合| 美女免费精品高清毛片在线视| 国产成人精品综合久久久| 欧美一级视频免费观看| 欧美激情伊人| 91麻豆精品国产综合久久久| 国产一区二区精品在线观看| 香蕉视频久久| 国产视频一区二区在线观看| 精品国产一区二区三区久| 沈樵在线观看福利| 精品国产一区二区三区免费| 免费毛片播放| 深夜做爰性大片中文| 精品视频在线观看免费| 精品国产一区二区三区久久久蜜臀 | 99热精品在线| 韩国毛片 免费| 国产激情视频在线观看| 欧美电影免费看大全| 日韩一级黄色大片| 国产伦精品一区二区三区无广告 | 国产精品自拍在线| 欧美日本免费| 韩国三级一区| 国产福利免费观看| 国产成a人片在线观看视频 | 韩国三级一区| 精品国产一区二区三区久| 九九久久国产精品大片| 国产不卡福利| 九九精品影院| 精品国产一区二区三区国产馆| 亚洲精品影院| 国产网站免费观看| 久久99爰这里有精品国产| 精品国产香蕉伊思人在线又爽又黄| 成人影院一区二区三区| 亚洲第一页色| 999精品视频在线| 青草国产在线观看| 午夜精品国产自在现线拍| 国产原创视频在线| 国产精品自拍亚洲| 午夜在线亚洲| 亚洲精品久久玖玖玖玖| 国产精品自拍一区| 99色视频在线观看| 国产麻豆精品免费密入口| 欧美激情一区二区三区在线| 国产国语对白一级毛片| 四虎久久精品国产| 成人影院久久久久久影院| 亚欧视频在线| 精品久久久久久免费影院| 欧美一区二区三区性| 日本在线播放一区| 精品国产一区二区三区精东影业| 欧美a级片免费看| 欧美夜夜骑 青草视频在线观看完整版 久久精品99无色码中文字幕 欧美日韩一区二区在线观看视频 欧美中文字幕在线视频 www.99精品 香蕉视频久久 | 色综合久久久久综合体桃花网| 99久久精品国产免费| 精品国产香蕉在线播出| 欧美日本免费| 久久国产精品自由自在| 国产国产人免费视频成69堂| 国产成人啪精品| 亚飞与亚基在线观看| 国产成人精品一区二区视频| 沈樵在线观看福利| 韩国毛片免费大片| 久草免费在线视频| 国产亚洲免费观看| 亚洲精品久久玖玖玖玖| 成人av在线播放| 国产一区二区精品久久91| 亚洲不卡一区二区三区在线| 亚洲 欧美 成人日韩| 九九久久国产精品| 日韩免费在线| 青青久久精品| 成人免费观看的视频黄页| 亚洲精品永久一区| 高清一级毛片一本到免费观看| 国产成人精品一区二区视频| 欧美激情一区二区三区在线| 中文字幕97| 国产网站免费视频| 99久久精品国产国产毛片| 日韩免费在线视频| 成人免费观看的视频黄页| 日本免费乱理伦片在线观看2018| 国产成a人片在线观看视频| 97视频免费在线观看| 欧美激情一区二区三区在线| 香蕉视频久久| 国产视频网站在线观看| 99久久精品国产高清一区二区| 亚洲www美色| 欧美激情伊人| 日日夜夜婷婷| 欧美日本免费| 香蕉视频亚洲一级| 你懂的在线观看视频| 黄色免费三级| 欧美1区| 黄视频网站免费观看| 欧美18性精品| 国产麻豆精品免费视频| 国产亚洲精品aaa大片| 国产美女在线一区二区三区| 欧美激情一区二区三区视频 | 成人免费观看男女羞羞视频| 精品视频在线观看视频免费视频 | 日本免费看视频| 日韩avdvd| 成人免费福利片在线观看| 精品视频一区二区| 国产美女在线一区二区三区| 青青青草影院| 欧美一区二区三区性| 国产91精品系列在线观看| 国产精品免费精品自在线观看| 日韩在线观看免费完整版视频| 黄色免费三级| 你懂的日韩| 日韩男人天堂| 欧美激情一区二区三区视频高清| 91麻豆精品国产综合久久久| 免费国产在线视频| 精品国产一级毛片| 国产视频在线免费观看| 欧美激情一区二区三区视频高清| 精品视频在线观看免费 | 成人免费高清视频|