远程过程调用协议gRPC及在go环境下的使用

1. 远程过程调用协议

1.1 定义

  远程过程调用(Remote Procedure Call,PRC是一种进程间通信技术,它使得程序可以像调用本地函数一样调用远程服务器上的函数。RPC 屏蔽了底层的通信细节,让开发者能够更专注于业务逻辑,而无需关心网络编程的复杂性。其流程如下:
在这里插入图片描述

1.2 grpc

  gRPC是由 Google 开发的一个高性能、开源的 RPC 框架,基于 HTTP/2 协议并使用Protocol Buffers(protobuf)作为接口描述语言。

2. Protocol Buffers

  Protocol Buffers(Protobuf)是由Google开发的一种语言中立、平台中立、可扩展的序列化数据格式。它用于在不同的程序和系统之间进行高效的数据交换。Protobuf 允许开发者定义数据结构(消息)并自动生成代码来处理这些数据结构的序列化和反序列化。Protocol Buffers具有以下优点:

  • 高效性:Protobuf 使用紧凑的二进制格式,比传统的文本格式(如 XML 或 JSON)更节省空间和带宽,序列化和反序列化速度快,适合高性能需求的场景。
  • 跨语言支持:Protobuf 支持多种编程语言,包括但不限于 C++, Java, Python, Go, C#, Ruby, JavaScript 等。这种跨语言特性使得它在异构系统中非常有用。
  • 可扩展性:Protobuf 允许数据结构的灵活扩展,新的字段可以添加而不会破坏已有的数据格式。这种向后兼容性使得在系统升级和维护过程中更容易处理。
  • 自动生成代码:开发者定义数据结构的.proto文件后,可以使用 protoc 编译器生成目标语言的代码,减少手工编码的错误和工作量。
2.1 基础语法

  Protobuf的定义文件.proto一般包含语法版本声明、包声明、消息定义和服务定义四个部分。其基本结构如下:

syntax = "proto3";
option go_package="./project/generated"
package example;
message MessageName {//定义消息
  // 字段定义
}
service Greeter { //定义服务
}
  • 语法版本声明:使用syntax指定.proto文件的语法版本,一般常用的语法版本有:proto2proto3
  • 包声明:package关键字用于指定代码的命名空间。
  • optiongo_pacakge指定了生成的Go代码应该放置的路径名。go_package中至少要包含一个/.
  • 消息定义:使用message关键字声明消息类型。消息类型中存放数据结构。一个消息类型包含一个或多个字段,每个字段有一个唯一的编号。举例如下:
message Person {
  string name = 1;   // 类型、字段名及其唯一编号。
  int32 id = 2;
  string email = 3;
}
  • 服务定义:使用service关键字声明服务类型,服务用于定义 RPC 服务接口。举例如下:
service Greeter {
  rpc SayHello (HelloRequest) returns (HelloReply);
}
message HelloRequest {
  string name = 1;
}
message HelloReply {
  string message = 1;
}

3. go环境下的grpc案例

3.1 安装相关的包

先在go语言的project中安装grpc相关的包,具体命令如下

go get google.golang.org/grpc
go get google.golang.org/protobuf
go get google.golang.org/grpc/cmd/protoc-gen-go-grpc@latest
go get google.golang.org/protobuf/cmd/protoc-gen-go@latest
3.2 编写helloworld.proto文件

如果go环境使用的是GoLand等,需要先安装protobuf插件。具体过程这里不在赘述。

syntax = "proto3";
option go_package="./generated";
package helloworld;
service Greeter {
  rpc SayHello (HelloRequest) returns (HelloReply);
}
message HelloRequest {
  string name = 1;
}
message HelloReply {
  string message = 1;
}

接着使用protoc命令将helloworld.proto文件编译成go语言的源代码。具体命令如下:

 protoc --go_out=. --go-grpc_out=. helloworld.proto

其中:
--go_out指定生成Go代码的输出目录。
--go-grpc_out指定生成Go gRPC代码的输出目录。

3.3 客户端/服务端代码

服务端代码server.go

package main
import (
	"context"
	"google.golang.org/grpc"
	pb "grpc_test/proto/generated" //引用protoc生成的go package
	"log"
	"net"
)
type server struct {
	pb.UnimplementedGreeterServer
}
//实现sayhello函数
func (s *server) SayHello(ctx context.Context, req *pb.HelloRequest) (*pb.HelloReply, error) {
	return &pb.HelloReply{Message: "Hello, " + req.Name}, nil
}
func main() {
	lis, err := net.Listen("tcp", ":50051")
	if err != nil {
		log.Fatalf("failed to listen: %v", err)
	}
	s := grpc.NewServer()
	pb.RegisterGreeterServer(s, &server{})
	if err := s.Serve(lis); err != nil {
		log.Fatalf("failed to serve: %v", err)
	}
}

客户端代码client.go

package main
import (
	"context"
	"flag"
	"log"
	"time"
	"google.golang.org/grpc"
	"google.golang.org/grpc/credentials/insecure"
	pb "grpc_test/proto/generated"
)
const (
	defaultName = "world"
)
var (
	addr = flag.String("addr", "localhost:50051", "the address to connect to")
	name = flag.String("name", defaultName, "Name to greet")
)
func main() {
	flag.Parse()
	// Set up a connection to the server.
	conn, err := grpc.NewClient(*addr, grpc.WithTransportCredentials(insecure.NewCredentials()))
	if err != nil {
		log.Fatalf("did not connect: %v", err)
	}
	defer conn.Close()
	c := pb.NewGreeterClient(conn)
	// Contact the server and print out its response.
	ctx, cancel := context.WithTimeout(context.Background(), time.Second)
	defer cancel()
	r, err := c.SayHello(ctx, &pb.HelloRequest{Name: *name})
	if err != nil {
		log.Fatalf("could not greet: %v", err)
	}
	log.Printf("Greeting: %s", r.GetMessage())
}

这里完整的代码结构如下:
在这里插入图片描述
然后分别server.goclient.go文件(如果是在GoLand中,这两个文件要分别运行在两个终端中),可以看到如下结果:

2024/06/25 20:51:17 Greeting: Hello, world

参考资料

  1. https://blog.csdn.net/m0_68949064/article/details/123974867

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/756933.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

C语言力扣刷题4——删除链表的倒数第 N 个结点[双指针],只遍历一遍

力扣刷题4——删除链表的倒数第 N 个结点[双指针] 一、博客声明二、题目描述三、解题思路1、思路说明 四、解题代码(附注释) 一、博客声明 找工作逃不过刷题,为了更好的督促自己学习以及理解力扣大佬们的解题思路,开辟这个系列来记…

“用友审批,兴业付款”,YonSuite让企业资金调拨更高效

随着企业规模的扩大和经营的不断深入,企业的分支机构分布各地,在多银行多账户的资金管理上面临着诸多挑战。传统的资金管理方式往往效率低下、风险较高,难以满足企业快速发展的需求。 为了解决这些痛点,用友网络科技股份有限公司…

MathType2024最新破解版在哪里可以下载?

在当今科技日益发展的时代,我们每个人都可能遇到需要在电子文档、网页或其他平台中输入复杂数学公式的情况。这时,一个强大且易用的数学公式编辑器就成了我们迫切需要的工具。而MathType,作为一款专业、精准的数学公式编辑器,无疑…

Redis为什么设计多个数据库

​关于Redis的知识前面已经介绍过很多了,但有个点没有讲,那就是一个Redis的实例并不是只有一个数据库,一般情况下,默认是Databases 0。 一 内部结构 设计如下: Redis 的源码中定义了 redisDb 结构体来表示单个数据库。这个结构有若干重要字段,比如: dict:该字段存储了…

人生最有力,最棒的十句话!

人生最有力,最棒的十句话 1、允许一切事发生,所有一切发生的事不是你能阻挡了的,你接受,他也发生,你不接受,他也发生,你还不如坦然面对接受现实。 2、你焦虑的时候千万不要躺着啥也不干&#xf…

机器人控制系列教程之关节空间运动控制器搭建(1)

机器人位置控制类型 机器人位置控制分为两种类型: 关节空间运动控制—在这种情况下,机器人的位置输入被指定为一组关节角度或位置的向量,这被称为机器人的关节配置,记作q。控制器跟踪一个参考配置,记作 q r e f q_{re…

2SK241 LTSpice模型及仿真

2SK241是东芝生产的一款NMOS,早已停产,但是在收音机圈子里还是有很多死忠粉,所以在淘宝上也是一堆打磨改标的假货。 言归正传,在矿坛上找到了2SK241的模型: .model M2SK241bottom NMOS(Level1 Rd1 Rs10 Rg50 Kp8mV…

《数据结构与算法基础 by王卓老师》学习笔记——类C语言有关操作补充

1.元素类型说明 2.数组定义 3.C语言的内存动态分配 4..C中的参数传递 5.传值方式 6.传地址方式 例子

CICD持续集成(Jenkins+Git+Gogs)

1.Jenkins Jenkins 是一个开源的、用于构建和自动化软件开发流程的持续集成和交付工具。它提供了一个可扩展的平台,用于构建、测试和部署软件项目。通过使用 Jenkins,开发团队可以实现持续集成和交付,自动化构建和测试过程,提高软…

C++Primer Plus 第十四章代码重用:14.4.7 成员模板

CPrimer Plus 第十四章代码重用 提示:这里可以添加系列文章的所有文章的目录,目录需要自己手动添加 例如:CPrimer Plus 第十四章代码重用:14.4.7 成员模板 提示:写完文章后,目录可以自动生成,如…

【算法训练记录——Day37】

Day37——贪心Ⅴ 1.leetcode_56合并区间 1.leetcode_56合并区间 思路:排序,如果重叠,更新right 为max(right, curVal), 不重叠就加入res,需要单独考虑最后一次,因为每次都是在下一次遍历开始时判断是否加入res,因此 当…

基于Java的广场舞团管理系统

你好呀,我是计算机学姐码农小野!如果有相关需求,可以私信联系我。 开发语言:Java 数据库:MySQL 技术:Java技术,B/S架构 工具:MyEclipse,MySQL 系统展示 首页 社团活…

AI是如何与快充技术结合的?

针对AI技术在快充领域的运用,我们可以进一步深入探讨AI如何与快充技术结合,提升充电效率和用户体验。以下是一些具体的AI技术在快充领域的应用场景: 一、智能充电算法 学习充电模式:AI算法可以学习用户的充电习惯,比…

微服务中的Feign远程调用

Feign的个人理解 Feign在英文中是“装”的意思,但在微服务中他是远程调用的一种方式,我的理解是:他替代了RestTemplateNacos中的URL编码的方式,显得很高大上,所以很装:(声明式事务,只…

端口扫描攻击检测及防御方案

端口扫描数据一旦落入坏人之手,可能会成为更大规模恶意活动的一部分。因此,了解如何检测和防御端口扫描攻击至关重要。 端口扫描用于确定网络上的端口是否开放以接收来自其他设备的数据包,这有助于网络安全团队加强防御。但恶意行为者也可以…

AI Prompt 提示词编写公式

自 OpenAI 的 ChatGPT 横空出世至今,各种 AI 大模型百花齐放、百家争鸣。按照用途可以分为两类: 对话类:即通过文字、语音、图片或者视频输入来给模型下达指令,然后模型按照指令以文字的形式将回答输出给用户;生成类&…

Web缓存代理和CDN 内容分发网络

目录 1.WEB缓存代理 1.1 WEB缓存代理作用 1.2 常见WEB缓存代理 1.3 Nginx 配置 缓存代理 2. CDN内容分发网络 1.WEB缓存代理 1.1 WEB缓存代理作用 存储一些之前给访问过的,且可能要被再次访问的静态网页资源对象,使客户端可以直接从缓存代理服务器…

钡铼BL104智慧环保多个485采集转MQTT无线传输

PLC物联网关BL104是一款专为工业环境设计的先进协议转换网关,其集成了钡铼智能技术和环保多个485采集转MQTT无线传输功能,为工业控制系统提供了高效的数据采集、传输和管理解决方案。 技术规格与功能特点 PLC物联网关BL104采用钡铼智能技术&#xff0c…

OpenCV学习之cv2.imshow()函数

OpenCV学习之cv2.imshow()函数 一、简介 cv2.imshow 是 OpenCV 库中用于显示图像的基本函数之一。在图像处理和计算机视觉的过程中,使用该函数可以快速预览处理后的图像,便于调试和结果展示。 二、基本语法 cv2.imshow(WindowName, Imgmat)三、参数说…

队列的相关知识

目录 创建 初始化 销毁 头插 尾删 取出头 取出尾 数字个数 判空 队列的性质与特征 性质:一种先进先出的线性表 特征:FIFO(先进先出) 实现:用数组和链表的都可以 例子:在生产者消费者模型用到了…