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

重學(xué)Go語(yǔ)言之如何開(kāi)發(fā)RPC應(yīng)用

 更新時(shí)間:2023年09月04日 15:33:13   作者:程序員讀書(shū)  
這篇文章主要為大家詳細(xì)介紹了在Go語(yǔ)言中如何構(gòu)建RPC應(yīng)用,文中的示例代碼講解詳細(xì),具有一定的學(xué)習(xí)價(jià)值,感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下

對(duì)RPC的理解

什么是RPC

RPCRemote Procedure Call的縮寫(xiě),中文譯為遠(yuǎn)程過(guò)程調(diào)用,通俗一點(diǎn)來(lái)說(shuō)就是通過(guò)網(wǎng)絡(luò)調(diào)用部署在遠(yuǎn)程服務(wù)器上的函數(shù)或方法,如下圖所示:

RPC與HTTP

談到RPC,往往會(huì)讓人聯(lián)想到另一種調(diào)用遠(yuǎn)程服務(wù)的方式:HTTP。

那為什么有了HTTP,還需要RPC呢?

其實(shí)使用RPC在不同主機(jī)進(jìn)程間通訊的時(shí)間要早于HTTP出現(xiàn)的時(shí)間。

因此,應(yīng)該這么問(wèn):既然有了RPC為何還需要HTTP呢?

我們知道,任何網(wǎng)絡(luò)應(yīng)用程序之間通訊都是基于TCP協(xié)議(當(dāng)然可以是UDP)。

TCP是傳輸層協(xié)議,其作用之一便是將從網(wǎng)絡(luò)層接收的數(shù)據(jù)傳給應(yīng)用層。

HTTP是一個(gè)應(yīng)用層協(xié)議,HTTP協(xié)議規(guī)定一條HTTP報(bào)文由請(qǐng)求行、請(qǐng)求頭和消息體組成,因此每條HTTP報(bào)文都有固定的格式。

使用RPC的方式進(jìn)行通訊時(shí),傳輸層依然是TCP協(xié)議,但是應(yīng)用層則需要通訊雙方約定好數(shù)據(jù)格式,相當(dāng)于自定義一個(gè)應(yīng)用層協(xié)議,因此RPC有各種不同的實(shí)現(xiàn),并沒(méi)有統(tǒng)一的規(guī)范。

gRPC框架的使用

gRPC是一個(gè)高性能開(kāi)源的RPC框架,支持Go,C++,Java,PHP,Ruby,Python等不同編程語(yǔ)言。

圖片來(lái)自于grpc官網(wǎng)

Protocol Buffers

Protocol BuffersGoogle開(kāi)發(fā)的一種與語(yǔ)言、平臺(tái)無(wú)關(guān)的數(shù)據(jù)序列化機(jī)制,這種機(jī)制由幾個(gè)部分組成:

  • protoc編譯器,用于編譯.proto文件。
  • .proto為后綴的IDL聲明文件,用于定義一個(gè)RPC服務(wù)。
  • 底層支持通訊并進(jìn)行編碼與解碼的庫(kù)。

Protocal Buffers有以下幾個(gè)特征:

  • Protocal BuffersJSON類(lèi)似,用于序列化數(shù)據(jù),不過(guò)與比于JSON其體積更小,因此傳輸也更快。
  • 支持多種編程語(yǔ)言。
  • 可以非??焖賯鬏斉c解析。

.proto文件

.proto文件用于聲明使用gRPC進(jìn)行通訊服務(wù)名稱、請(qǐng)求數(shù)據(jù)類(lèi)型、順序與響應(yīng)數(shù)據(jù)類(lèi)型、順序等信息。

.proto文件的第一行必須是:

syntax = "proto3";

如果沒(méi)有聲明為proto3,編譯器會(huì)以proto2的語(yǔ)法解析.proto文件。

message關(guān)鍵字用于定義一個(gè)消息類(lèi)型:

message SearchRequest {
  string query = 1;
  int32 page_number = 2;
  int32 results_per_page = 3;
}

同一個(gè).proto文件里可以定義多個(gè)消息類(lèi)型:

message SearchRequest {
  string query = 1;
  int32 page_number = 2;
  int32 results_per_page = 3;
}
message SearchResponse {
	//...
}

service關(guān)鍵字用于聲明一個(gè)服務(wù),格式如下:

service Search {
  rpc Search (SearchRequest) returns (SearchReply) {}
}

編寫(xiě)好的.proto文件,要使用protoc編譯進(jìn)行編譯,生成目標(biāo)語(yǔ)言的代碼。

開(kāi)發(fā)工具安裝

當(dāng)然在開(kāi)發(fā)之前,除了安裝Go語(yǔ)言環(huán)境外,還需要安裝以下幾個(gè)工具:

  • protoc
  • protoc-gen-go
  • protoc-gen-go-grpc

protoc

protoc.proto文件的編譯器,其作用是將.proto文件中聲明的信息轉(zhuǎn)為目標(biāo)語(yǔ)言的代碼。

protoc可以從以下地址下載:https://github.com/protocolbuffers/protobuf/releases

下載后,將其配置到PATH路徑下即可。

protoc-gen-go與protoc-gen-go-grpc

如果要protoc編譯器可以生成go代碼,還需要安裝protoc-gen-goprotoc-gen-go-grpc插件。

protoc-gen-goprotoc-gen-go-grpc插件用于生成go以及grpc代碼,這兩個(gè)插件由protoc命令調(diào)用。

使用go install將兩個(gè)命令安裝到GOPATH/bin目錄下:

$?go?install?google.golang.org/protobuf/cmd/protoc-gen-go@latest
$?go?install?google.golang.org/grpc/cmd/protoc-gen-go-grpc@latest

GOPATH/bin目錄要配置到PATH目錄下

案例實(shí)戰(zhàn)

安裝了相關(guān)工具以及了解了Protocol Buffers.proto文件,下面我們通過(guò)一個(gè)實(shí)際案例來(lái)了解RPC應(yīng)用的開(kāi)發(fā)。

創(chuàng)建項(xiàng)目

首先執(zhí)行以下命令創(chuàng)建一個(gè)Go項(xiàng)目:

$?mkdir?test
$?cd?test
$?go?mod?init?test

定義.proto文件

gRPC開(kāi)發(fā)RPC應(yīng)用的第一件事就是定義.proto文件,在這個(gè)項(xiàng)目中,我們?cè)?code>user目錄下創(chuàng)建user.proto文件:

$?mkdir?user
$?touch?user.proto

user.proto文件中輸入以下代碼:

syntax = "proto3";
option go_package = "test/user";
//定義兩個(gè)服務(wù)
service User {
    rpc GetUser (UserId) returns (UserInfoReply) {}
    rpc AddUser (AddUserRequest) returns(UserId){}
}
message UserId {
    int32 id = 1;
}
message UserInfoReply {
    int32 id = 1;
    string name = 2;
    string email = 3;
}
message AddUserRequest {
    string name = 1;
    string email = 2;
}

編譯.proto文件:

.proto文件編寫(xiě)完成后,執(zhí)行protoc命令編譯該文件,生成目標(biāo)語(yǔ)言代碼:

protoc?--go_out=.?--go_opt=paths=source_relative?\
????--go-grpc_out=.?--go-grpc_opt=paths=source_relative?\
????user/user.proto

編譯成功后會(huì)在user目錄生成user.pb.gouser_grpc.pb.go兩個(gè)文件。

編寫(xiě)服務(wù)端代碼

下面是gRPC應(yīng)用的服務(wù)端代碼:

package?main
import?(
?"context"
?"log"
?"net"
?"test/user"
?"google.golang.org/grpc"
)
type?userServer?struct?{
?user.UnimplementedUserServer
}
//[1]
func?(s?*userServer)?GetUser(ctx?context.Context,?in?*user.UserId)?(*user.UserInfoReply,?error)?{
?log.Printf("請(qǐng)求用戶id為:?%d",?in.GetId())
?return?&user.UserInfoReply{Id:?1,?Name:?"程序員讀書(shū)",?Email:?"test@163.com"},?nil
}
//[2]
func?(s?*userServer)?AddUser(ctx?context.Context,?in?*user.AddUserRequest)?(*user.UserId,?error)?{
?log.Printf("你要添加的用戶名稱為:?%s,郵箱為:%s",?in.GetName(),?in.GetEmail())
?return?&user.UserId{Id:?2},?nil
}
func?main()?{
?listen,?err?:=?net.Listen("tcp",?":50051")
?if?err?!=?nil?{
??log.Fatalf("failed?to?listen:?%v",?err)
?}
?s?:=?grpc.NewServer()
?user.RegisterUserServer(s,?&userServer{})
?log.Printf("server?listening?at?%v",?listen.Addr())
?if?err?:=?s.Serve(listen);?err?!=?nil?{
??log.Fatalf("failed?to?serve:?%v",?err)
?}
}

在上面的代碼中,主要完成以下幾件事:

  • 創(chuàng)建一個(gè)監(jiān)聽(tīng)器監(jiān)聽(tīng)50051端口
  • 通過(guò)grpc.NewServer()創(chuàng)建RPC服務(wù)器,將服務(wù)對(duì)象userServer綁定服務(wù)器
  • 將監(jiān)聽(tīng)器傳給RPC服務(wù)器以啟動(dòng)服務(wù)。

編寫(xiě)客戶端代碼

package?main
import?(
?"context"
?"log"
?"os"
?"time"
?"test/user"
?"google.golang.org/grpc"
)
func?main()?{
?conn,?err?:=?grpc.Dial("localhost:50051",?grpc.WithInsecure(),?grpc.WithBlock())
?if?err?!=?nil?{
??log.Fatalf("did?not?connect:?%v",?err)
?}
?defer?conn.Close()
??ctx,?cancel?:=?context.WithTimeout(context.Background(),?time.Second)
?defer?cancel()
?u?:=?user.NewUserClient(conn)
?userInfo,?err?:=?u.GetUser(ctx,?&user.UserId{Id:?1})
?if?err?!=?nil?{
??log.Fatalf("user?nto?found:?%v",?err)
?}
??userId,err?:=?u.AddUser(ctx,&user.AddUserRequest{Name:"test",Email:"test@test.com"})
}

小結(jié)

本文主要介紹了使用gRPC與Go語(yǔ)言進(jìn)行RPC應(yīng)用的開(kāi)發(fā),總結(jié)起來(lái)就是以下幾點(diǎn):

  • 通過(guò)網(wǎng)絡(luò)調(diào)用遠(yuǎn)程主機(jī)的函數(shù),稱為RPC。
  • gRPC是一個(gè)實(shí)現(xiàn)RPC的框架,支持多種編程語(yǔ)言。
  • gRpc使用.proto文件描述一個(gè)RPC服務(wù),并用protoc命令生成目標(biāo)語(yǔ)言的代碼。

到此這篇關(guān)于重學(xué)Go語(yǔ)言之如何開(kāi)發(fā)RPC應(yīng)用的文章就介紹到這了,更多相關(guān)Go RPC內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • Go語(yǔ)言中的range用法實(shí)例分析

    Go語(yǔ)言中的range用法實(shí)例分析

    這篇文章主要介紹了Go語(yǔ)言中的range用法,實(shí)例分析了range的功能與使用技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下
    2015-02-02
  • 解決golang處理http response碰到的問(wèn)題和需要注意的點(diǎn)

    解決golang處理http response碰到的問(wèn)題和需要注意的點(diǎn)

    這篇文章主要介紹了解決golang處理http response碰到的問(wèn)題和需要注意的點(diǎn),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2020-12-12
  • golang 中signal包的Notify用法說(shuō)明

    golang 中signal包的Notify用法說(shuō)明

    這篇文章主要介紹了golang 中signal包的Notify用法說(shuō)明,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2021-03-03
  • vscode插件設(shè)置之Golang開(kāi)發(fā)環(huán)境配置全過(guò)程

    vscode插件設(shè)置之Golang開(kāi)發(fā)環(huán)境配置全過(guò)程

    go語(yǔ)言開(kāi)發(fā)選擇vscode作為IDE工具也是一個(gè)不錯(cuò)的選擇,下面這篇文章主要給大家介紹了關(guān)于vscode插件設(shè)置之Golang開(kāi)發(fā)環(huán)境配置的相關(guān)資料,文中通過(guò)圖文介紹的非常詳細(xì),需要的朋友可以參考下
    2022-12-12
  • Go語(yǔ)言學(xué)習(xí)之函數(shù)的定義與使用詳解

    Go語(yǔ)言學(xué)習(xí)之函數(shù)的定義與使用詳解

    這篇文章主要為大家詳細(xì)介紹Go語(yǔ)言中函數(shù)的定義與使用,文中的示例代碼講解詳細(xì),對(duì)我們學(xué)習(xí)Go語(yǔ)言有一定幫助,需要的可以參考一下
    2022-04-04
  • Go庫(kù)text與template包使用示例詳解

    Go庫(kù)text與template包使用示例詳解

    這篇文章主要為大家介紹了Go庫(kù)text與template包使用示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-12-12
  • 基于原生Go語(yǔ)言開(kāi)發(fā)一個(gè)博客系統(tǒng)

    基于原生Go語(yǔ)言開(kāi)發(fā)一個(gè)博客系統(tǒng)

    這篇文章主要為大家詳細(xì)介紹了如何基于原生Go語(yǔ)言開(kāi)發(fā)一個(gè)簡(jiǎn)單的博客系統(tǒng),文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下
    2024-02-02
  • 深入分析Go?實(shí)現(xiàn)?MySQL?數(shù)據(jù)庫(kù)事務(wù)

    深入分析Go?實(shí)現(xiàn)?MySQL?數(shù)據(jù)庫(kù)事務(wù)

    本文深入分析了Go語(yǔ)言實(shí)現(xiàn)MySQL數(shù)據(jù)庫(kù)事務(wù)的原理和實(shí)現(xiàn)方式,包括事務(wù)的ACID特性、事務(wù)的隔離級(jí)別、事務(wù)的實(shí)現(xiàn)方式等。同時(shí),本文還介紹了Go語(yǔ)言中的事務(wù)處理機(jī)制和相關(guān)的API函數(shù),以及如何使用Go語(yǔ)言實(shí)現(xiàn)MySQL數(shù)據(jù)庫(kù)事務(wù)。
    2023-06-06
  • golang關(guān)閉chan通道的方法示例

    golang關(guān)閉chan通道的方法示例

    在go語(yǔ)言中,通道(channel)是一個(gè)非常重要的概念,通道提供了一種在不同 goroutine 之間安全地傳遞數(shù)據(jù)的方式,在本文中,我們將討論如何關(guān)閉通道以及在關(guān)閉通道時(shí)需要考慮的事項(xiàng),需要的朋友可以參考下
    2024-02-02
  • 使用Golang簡(jiǎn)單實(shí)現(xiàn)七牛圖片處理API

    使用Golang簡(jiǎn)單實(shí)現(xiàn)七牛圖片處理API

    本文給大家實(shí)現(xiàn)的是使用Golang簡(jiǎn)單實(shí)現(xiàn)七牛圖片處理API的方法和步驟,基于PIPE庫(kù)實(shí)現(xiàn)的,非常的實(shí)用,有需要的小伙伴可以參考下
    2016-08-08

最新評(píng)論