Go-grpc 实现

什么是grpc和protobuf

grpc

​ grpc是一个Google开源的高性能、开源和通用的RPC框架,面向移动和HTTP/2设计。目前提供C、Java和Go语言版本, 分别是grpc, grpc-java 和 grpc-go, 其中C语言版本又支持C , C++,Node.js, Python, Ruby, Object-C, PHP, C#

grpc协议使用的序列化程序不是json 、xml 等, 而是使用的protobuf序列化及反序列化

protobuf

  • 习惯使用json、xml交互数据的人,大多没有听说过 Protocol Buffer
  • Protocol Buffer其实是Google出品的一种轻量&高效的结构化数据存储格式,性能要比json、xml强很多
  • 目前主流使用的protobuf3

go get -d google.golang.org/protobuf/cmd/protoc-gen-go

安装好protoc执行程序后 写入.proto文件

syntax = "proto3";
option go_package = ".;proto";

service Greeter{
  rpc SayHello (HelloRequest) returns (HelloReply);
}

message HelloRequest {
  string name = 1;
}

message HelloReply {
  string message = 1;

}

写完之后 执行命令生成.go文件protoc -I . first.proto --go_out=plugins=grpc:.

  • Server端代码
package main

import (
	"context"
	"log"
	"net"

	"google.golang.org/grpc"
	"google.golang.org/grpc/reflection"

	pb "AwesomeMicroPro/MicroProject/proto"
)

// server is used to implement pb.GreeterServer .
type server struct {
	pb.UnimplementedGreeterServer
}


// SayHello 方法接受远程调用方法
func (s *server) SayHello(ctx context.Context, request *pb.HelloRequest) (*pb.HelloReply, error) {
	log.Println(request.Name + "remote 1 first")
	return &pb.HelloReply{Message: "hello " + request.Name}, nil
}


func main() {
	g := grpc.NewServer()
	reflection.Register(g)
	pb.RegisterGreeterServer(g, &server{})
	listen, err := net.Listen("tcp", "127.0.0.1:8080")
	if err != nil {
		panic("Listen Port Failed"+err.Error())
	}

	err = g.Serve(listen)
	if err != nil {
		panic("Server Enable Failed"+err.Error())
	}

}

  • Client端代码
package main

import (
	"context"
	"google.golang.org/grpc"
	"log"
	"os"

	pb "AwesomeMicroPro/MicroProject/proto"
)

const (
	address = "127.0.0.1:8080"
)

func main() {
	// 拨号连接Server
	conn, err := grpc.Dial(address, grpc.WithInsecure())
	if err != nil {
		panic("Connect grpc Server Failed" + err.Error())
	}
	// 最后关闭连接
	defer conn.Close()

	client := pb.NewGreeterClient(conn)
	name := "testing ***"
	if len(os.Args) > 1 {
		name = os.Args[1]
	}
	// 远程调用SayHello方法传入对应的值
	reply, err := client.SayHello(context.Background(), &pb.HelloRequest{Name: name})
	if err != nil {
		panic("fun remote execute failed" + err.Error())
	}

	log.Println(reply.Message)

}

rpc的四种模式

  • 简单模式
    • 简单模式最为传统 和熟悉的数据流模式没有任何区别,即客户端请求一次数据 服务端相应一个数据
  • 服务端数据流模式
    • 客户端发起一次请求 服务端连续不断的返回数据流 stream, 典型的例子就是客户端发起一个股票代码 服务端实时的将股票的信息返回到客户端
  • 客户端数据流模式
    • 客户端源源不断的向服务端发送数据流 ,发送结束后 服务端返回一个响应数据, 例子 : 物联网终端向服务器报告数据
  • 双向数据流模式
    • 客户端和服务端都可以发送数据流 、实时交互。 例子: 聊天

服务端数据流、客户端数据流、双向数据流 模式

stream.proto

syntax = "proto3";
option go_package = "./;proto";


service Greeter  {

  rpc GetStream (StreamReqData) returns (stream StreamResData) {} // 服务端流模式

  rpc PutStream (stream StreamReqData) returns (StreamResData) {} // 客户端流模式

  rpc AllStream (stream StreamResData) returns (stream StreamReqData) {} // 双向流模式
}

message StreamReqData {

  string data = 1;
}
message StreamResData {
  string data = 2;
}

protoc -I . stream.proto --go_out=plugins=grpc:.

server.go

package main

import (
	"AwesomeMicroPro/stream_grpc_test/proto"
	"fmt"
	"google.golang.org/grpc"
	"log"
	"net"
	"sync"
	"time"
)

const PORT = ":50052"


var wg sync.WaitGroup

type server struct {
}

// GetStream 服务端流模式 源源不断的发送数据给客户端
func (s *server) GetStream(data *proto.StreamReqData, res proto.Greeter_GetStreamServer) error {

	for i := 0; i < 10; i++ {
		_ = res.Send(&proto.StreamResData{
			Data: fmt.Sprintf("%v", time.Now().Unix()),
		})
		time.Sleep(time.Second)
	}

	return nil
}

// PutStream 客户端流模式  源源不断的接受客户端的请求
func (s *server) PutStream(putStreamServer proto.Greeter_PutStreamServer) error {
	for i:=0; i< 10; i ++{
		recv, err := putStreamServer.Recv()
		if err != nil {
			log.Println("Recv Data Failed"+ err.Error())
			break
		}
		fmt.Println(recv)
	}
	return nil
}

func (s *server) AllStream(allstream proto.Greeter_AllStreamServer) error {
	wg.Add(2)
	go func() {
		for {
			recv, err := allstream.Recv()
			if err != nil {
				log.Println("接受数据失败")
				break
			}
			fmt.Println("服务端接收到的数据:"+recv.Data)
		}
		defer wg.Done()
	}()

	go func() {
		for  {
			err := allstream.Send(&proto.StreamReqData{
				Data: fmt.Sprintf("我是服务端%v", time.Now().Unix()),
			})
			if err != nil {
				log.Println("服务端发送数据失败" + err.Error())
				break
			}
			time.Sleep(time.Second)
		}
		wg.Done()
	}()

	wg.Wait()
	return nil
}

func main() {
	listen, err := net.Listen("tcp", PORT)
	if err != nil {
		log.Println("Listen tcp port failed" + err.Error())
	}
	newServer := grpc.NewServer()
	proto.RegisterGreeterServer(newServer, &server{})
	err = newServer.Serve(listen)
}

Client.go

package main

import (
	"AwesomeMicroPro/stream_grpc_test/proto"
	"context"
	"fmt"
	"google.golang.org/grpc"
	"log"
	"sync"
	"time"
)

var wg sync.WaitGroup

func main() {
	conn, err := grpc.Dial("localhost:50052", grpc.WithInsecure())
	if err != nil {
		log.Println("connect grpc service failed" + err.Error())
	}

	defer conn.Close()

	client := proto.NewGreeterClient(conn)
	// GetStream 服务端流模式  适合订阅监控某些信息
	stream, err := client.GetStream(context.Background(), &proto.StreamReqData{Data: "服务端数据流调用"})

	for i:=0; i < 10; i++{
		recv, err := stream.Recv()
		if err != nil {
			log.Println(err.Error())
		}
		log.Println(recv)
	}
	// 客户端流模式
	putStream, err := client.PutStream(context.Background())
	for i:=0; i < 10; i ++ {
		err := putStream.Send(&proto.StreamReqData{Data: "**da **" + string(i)})
		if err != nil {
			log.Println("err" + err.Error())
			break
		}
	}

	// 双向流模式
	allStream, err := client.AllStream(context.Background())
	wg.Add(2)
	go func() {
		for {
			recv, err := allStream.Recv()
			if err != nil {
				log.Println("接收数据失败")
				break
			}
			fmt.Println("客户端接收到的数据: " + recv.Data)
		}
		defer wg.Done()
		
	}()
	go func() {
		for {
			err := allStream.Send(&proto.StreamResData{Data: fmt.Sprintf("我是客户端 %v", time.Now().Unix())})
			if err != nil {
				log.Println("客户端发送数据失败"+err.Error())
				break
			}
			time.Sleep(time.Second)
		}
		wg.Done()
	}()
	wg.Wait()
}

posted @ 2022-02-18 16:09  听风走了八千里  阅读(373)  评论(0编辑  收藏  举报