gRPC 双向流(Bidirectional Streaming RPC)的使用方法

gRPC 是一个支持多种语言的高性能 RPC 框架,拥有丰富的 API 来简化服务端和客户端的开发过程。gRPC 支持四种 RPC 类型:Unary RPC、Server Streaming RPC、Client Streaming RPC 和 Bidirectional Streaming RPC。下面是双向流 API 的使用方法。

双向流 API 示例

在双向流中,客户端和服务器之间可以同时发送和接收消息。以下是一个双向流的完整示例,展示了客户端和服务端之间的聊天功能。

1. 定义 Protobuf 文件

首先,定义一个 Protobuf 文件 chat.proto,描述服务和消息结构:

bash 复制代码
syntax = "proto3";

package chat;

service ChatService {
  // 双向流 RPC
  rpc Chat(stream ChatMessage) returns (stream ChatMessage);
}

message ChatMessage {
  string user = 1;
  string text = 2;
}
2. 编译 Protobuf 文件

使用 protoc 编译器生成 Go 代码:

bash 复制代码
protoc --go_out=paths=source_relative:. --go-grpc_out=paths=source_relative:. chat.proto

这会生成 chat.pb.go 文件,其中包含了定义的服务和消息。

3. 实现服务器

创建 server.go 文件,具体实现 ChatService 服务:

Go 复制代码
package main

import (
	"fmt"
	"io"
	"log"
	"net"

	pb "path/to/your/proto/package"
	"google.golang.org/grpc"
)

type server struct {
	pb.UnimplementedChatServiceServer
}

func (s *server) Chat(stream pb.ChatService_ChatServer) error {
	for {
		in, err := stream.Recv()
		if err == io.EOF {
			return nil
		}
		if err != nil {
			return err
		}
		log.Printf("Received message from %s: %s", in.User, in.Text)

		// Echo message back to client
		err = stream.Send(&pb.ChatMessage{User: "Server", Text: "Echo: " + in.Text})
		if err != nil {
			return err
		}
	}
}

func main() {
	listen, err := net.Listen("tcp", ":50051")
	if err != nil {
		log.Fatalf("failed to listen: %v", err)
	}

	s := grpc.NewServer()
	pb.RegisterChatServiceServer(s, &server{})

	log.Println("Server is running on port :50051")
	if err := s.Serve(listen); err != nil {
		log.Fatalf("failed to serve: %v", err)
	}
}
4. 实现客户端

创建 client.go 文件,用于连接服务器并发送和接收消息:

Go 复制代码
package main

import (
	"bufio"
	"context"
	"fmt"
	"log"
	"os"
	"time"

	pb "path/to/your/proto/package"
	"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()

	client := pb.NewChatServiceClient(conn)

	stream, err := client.Chat(context.Background())
	if err != nil {
		log.Fatalf("failed to create stream: %v", err)
	}

	waitc := make(chan struct{})

	go func() {
		reader := bufio.NewReader(os.Stdin)
		for {
			fmt.Print("Enter message: ")
			text, _ := reader.ReadString('\n')

			if err := stream.Send(&pb.ChatMessage{User: "Client", Text: text}); err != nil {
				log.Fatalf("failed to send message: %v", err)
				break
			}
		}
		stream.CloseSend()
	}()

	go func() {
		for {
			in, err := stream.Recv()
			if err == io.EOF {
				close(waitc)
				return
			}
			if err != nil {
				log.Fatalf("failed to receive message: %v", err)
				break
			}
			log.Printf("Received message from %s: %s", in.User, in.Text)
		}
	}()

	<-waitc
}
5. 运行示例

编译并运行服务器和客户端:

  • 编译服务器和客户端:

    Go 复制代码
    go build -o server server.go
    go build -o client client.go
  • 启动服务器:

    Go 复制代码
    ./server
  • 启动客户端:

    Go 复制代码
    ./client
    总结

    在这个例子中,服务器和客户端可以通过双向流(Bidirectional Streaming RPC)方式进行实时通信。客户端通过终端输入消息,并通过 gRPC 发送到服务器,服务器接收到消息后回显给客户端。这个示例展示了 gRPC 强大且灵活的流处理能力,在实际项目中可以根据需求进行扩展和修改。

Go 复制代码
import cycle not allowedgo list
could not import example.com/user/go-project/grpc/service2 (missing metadata for import of "example.com/user/go-project/grpc/service2")

这个错误提示说您在导入路径中循环引用或缺少某些必要的元数据。通常,这些问题可能是因为错误的导入路径配置、循环依赖或者项目结构问题。

步骤 1:确认 go.mod 和目录结构正确

确保 go.mod 文件在项目根目录中,并且包含正确的模块路径和依赖项。

假设项目结构为:

Go 复制代码
/home/user/go-project/grpc/
├── go.mod
├── go.sum
├── service1/
│   ├── chat.proto
│   ├── chat.pb.go
│   ├── chat_grpc.pb.go
│   ├── server.go
│   └── client.go
└── service2/
    ├── chat.proto
    ├── chat.pb.go
    ├── chat_grpc.pb.go
    ├── server.go
    └── client.go

确认 go.mod 文件

./grpc 目录下,初始化并正确配置 go.mod 文件:

Go 复制代码
cd /home/user/go-project/grpc
go mod init example.com/user/go-project/grpc

更新 go.mod 文件以包含必要的依赖,确保内容类似如下:

Go 复制代码
module example.com/user/go-project/grpc

go 1.17

require (
    google.golang.org/grpc v1.39.0
    google.golang.org/protobuf v1.26.0
)

运行以下命令来下载和更新所有依赖:

Go 复制代码
go mod tidy
相关推荐
Bruce_Liuxiaowei14 分钟前
SSH主机密钥验证失败(Host key verification failed)深度解析与解决方案
运维·网络·ssh
小豪GO!1 小时前
HTTPS原理
网络协议·http·https
星瞰物联1 小时前
融合北斗与天通卫星通信技术的堤坝水文监测卫星图传系统
网络·物联网·安全·系统架构
周杰伦_Jay2 小时前
【GRPC 和 HTTP】设计目标和底层实现
网络·网络协议·http
汤愈韬2 小时前
防火墙用户管理技术
网络协议·网络安全·huawei
木子欢儿2 小时前
Prometheus Blackbox域名SSL证书监控并设置AlertManager告警
网络·网络协议·ssl·prometheus
北京耐用通信3 小时前
解码协议迷雾:耐达讯自动化Profinet转Devicenet让食品包装称重模块“跨界对话”的魔法
人工智能·物联网·网络协议·自动化·信息与通信
猫天意3 小时前
【即插即用模块】AAAI2025 | 高频 + 空间感知!新 HS-FPN 让“极小目标”不再消失!SCI保二区争一区!彻底疯狂!!!
网络·人工智能·深度学习·学习·音视频
草根站起来3 小时前
SSL证书根证书没有二级根证书、三级根证书、四级根证书,SSL证书参数诈骗被迫使用了套牌贴牌非原厂SSL和证书
网络·网络协议·ssl
谢尔登3 小时前
HTTP 协议组成
网络·网络协议·http