grpc使用教程

准备

复制代码
1,安装grpc
go get -u google.golang.org/grpc

2, 安装go语言protobuf生成器protoc-gen-go
go get -u google.golang.org/protobuf/cmd/protoc-gen-go

3, 通过下面连接,找到合适版本并安装protoc工具,如windows选择 protoc-3.19.5-win64.zip 
https://github.com/protocolbuffers/protobuf/releases/tag/v3.19.5

注意: 将GOPATH下的bin目录以及protoc的bin目录配置在环境变量中否则会找不到相关命令

编写proto文件

复制代码
# server/proto/test.proto 文件
syntax = "proto3";

package test;
option go_package = "../proto";

service Greeter {
    rpc SayHello (HelloRequest) returns (HelloResponse) {}
}

message HelloRequest {
    string name = 1;
}

message HelloResponse {
    string message = 1;
}

生成protobuf

复制代码
cd server/proto
protoc --proto_path=. --go_out=plugins=grpc:. *

编写服务端程序

复制代码
# server/main.go
package main

import (
	"context"
	"google.golang.org/grpc"
	"log"
	"net"
	pb "server/proto"
)

type server struct {
}

func (s *server) SayHello(ctx context.Context, in *pb.HelloRequest) (*pb.HelloResponse, error) {
	return &pb.HelloResponse{
		Message: "hello " + in.Name,
	}, nil
}

func main() {
	listen, err := net.Listen("tcp", ":8080")
	if err != nil {
		log.Fatal("net.Listen err", err)
	}
	s := grpc.NewServer()
	pb.RegisterGreeterServer(s, &server{})

	if err = s.Serve(listen); err != nil {
		log.Fatal("s.Serve err", err)
	}
}

编写客户端

复制代码
# 先将server/proto 目录复制到 client/proto

# client/main.go
package main

import (
	pb "client/proto"
	"context"
	"google.golang.org/grpc"
	"google.golang.org/grpc/credentials/insecure"
	"log"
	"net"
	"net/http"
)

type server struct {
	httpServer http.Server
}

func (s *server) ServeHTTP(w http.ResponseWriter, r *http.Request) {
	if r.URL.Path == "/sayHello" {
	    _ = r.ParseForm()
		message := sayHello(r.Form.Get("name"))
		w.WriteHeader(http.StatusOK)
		w.Header().Set("Content-Type", "text/html")
		w.Write([]byte("<h1>" + message + "</h1>"))
	}
}

func sayHello(name string) string {
	conn, err := grpc.Dial("127.0.0.1:8080", grpc.WithTransportCredentials(insecure.NewCredentials()))
	if err != nil {
		log.Fatal("连接失败:", err)
	}
	defer conn.Close()
	client := pb.NewGreeterClient(conn)

	resp, err := client.SayHello(context.Background(), &pb.HelloRequest{Name: name})
	if err != nil {
		log.Fatal("调用失败:", err)
	}
	return resp.GetMessage()
}

func main() {
	listen, err := net.Listen("tcp", ":80")
	if err != nil {
		log.Fatal("net.Listen err", err)
	}

	s := &server{}
	s.httpServer.Handler = s
	if err = s.httpServer.Serve(listen); err != nil {
		log.Fatal("httpServer.Serve err", err)
	}
}

验证

复制代码
通过postman请求 http://127.0.0.1/sayHello 接口, 参数 name 为jack, 看到响应了 <h1>hello jack</h1>
相关推荐
民乐团扒谱机几秒前
脉冲在克尔效应下的频谱展宽仿真:原理与 MATLAB 实现
开发语言·matlab·光电·非线性光学·克尔效应
yuan199974 分钟前
基于扩展卡尔曼滤波的电池荷电状态估算的MATLAB实现
开发语言·matlab
Tony Bai6 分钟前
Go GUI 开发的“绝境”与“破局”:2025 年现状与展望
开发语言·后端·golang
豆浆whisky6 分钟前
Go分布式追踪实战:从理论到OpenTelemetry集成|Go语言进阶(15)
开发语言·分布式·golang
2401_860494707 分钟前
Rust语言高级技巧 - RefCell 是另外一个提供了内部可变性的类型,Cell 类型没办法制造出直接指向内部数据的指针,为什么RefCell可以呢?
开发语言·rust·制造
Tony Bai7 分钟前
【Go模块构建与依赖管理】08 深入 Go Module Proxy 协议
开发语言·后端·golang
浪裡遊8 分钟前
Next.js路由系统
开发语言·前端·javascript·react.js·node.js·js
程序员-小李9 分钟前
基于 Python + OpenCV 的人脸识别系统开发实战
开发语言·python·opencv
QX_hao9 分钟前
【Go】--文件和目录的操作
开发语言·c++·golang
卡提西亚10 分钟前
C++笔记-20-对象特性
开发语言·c++·笔记