go rpc

运用go标准库写一个rpc例子

服务端

go 复制代码
package main

import (
	"fmt"
	"net"
	"net/rpc"
)

//对象
type Hello struct {
}
//对象方法
func (h *Hello) HelloWorld(name string, resp *string) error {
	*resp = name + "你好"
	return nil
}

func main() {
	//1. 注册RPC服务,绑定对象方法
	err := rpc.RegisterName("hello", new(Hello))
	if err != nil {
		fmt.Println("rpc RegisterName error")
		return
	}

	//2. 初始化监听器
	listener, err := net.Listen("tcp", "127.0.0.1:8800")
	if err != nil {
		fmt.Println("net Listen error")
		return
	}
	defer listener.Close()
	
	//3.建立连接
	conn, err := listener.Accept()
	if err != nil {
		fmt.Println("listener Accept error")
		return
	}
	defer conn.Close()
	
	//4.将连接绑定rpc服务
	rpc.ServeConn(conn)
}

注意绑定的对象方法,必须有两个参数,第二个参数必须是"指针类型"为传出参数。方法只有一个error类型的返回值。

客户端

go 复制代码
package main

import (
	"fmt"
	"net/rpc"
)

func main() {
	//建立连接
	conn, err := rpc.Dial("tcp", "127.0.0.1:8800")
	if err != nil {
		fmt.Println("rpc Dial error")
		return
	}
	defer conn.Close()
	//调用远程函数
	var resp string
	err = conn.Call("hello.HelloWorld", "李白", &resp)
	if err != nil {
		fmt.Println("conn Call error")
		return
	}
	fmt.Println(resp)
}

注意:上面代码中rpc的远程函数调用采用的是golang中特有的序列化包gob,其他语音不能解析。所以,要进行不同语言之间的rpc通信可以采用通用的序列化方式,如json,protobuf

然后,在实际运用中都会对服务端和客户端的再进行封装:

protobuf添加rpc服务

此时使用的protobuf编译指令,将定义好的proto规则和rpc服务生成go代码,需要用到的是grpc

bash 复制代码
protoc --go_out=. *.proto
protoc --go-grpc_out=. *.proto

举例:

定义proto文件,然后用上面两个命令生成go代码

go 复制代码
syntax="proto3";

package pb;

//option go_package = "path;name"; path 表示生成的go文件的存放地址,会自动生成目录的
// name 表示生成的go文件所属的包名
option go_package=".;pb";

message HelloRequest {
  string requestName = 1;
}

message HelloResponse {
  string responseMsg = 1;
}

service Hello{
  rpc SayHello(HelloRequest) returns (HelloResponse);
}

服务端:

go 复制代码
package main

import (
	"context"
	"fmt"
	"google.golang.org/grpc"
	"net"
	"rpc_day01/pb"
)

type My struct {
	pb.UnimplementedHelloServer //必须嵌套这个
}

func (m *My) SayHello(c context.Context, req *pb.HelloRequest) (*pb.HelloResponse, error) {
	var resp pb.HelloResponse
	resp.ResponseMsg = req.RequestName + " hello !!"
	return &resp, nil
}

func main() {
	//1 初始化一个gpc对象
	grpcServer := grpc.NewServer()
	//2 注册服务
	pb.RegisterHelloServer(grpcServer, &My{})

	//3 设置监听,指定IP
	listener, err := net.Listen("tcp", "127.0.0.1:8801")
	if err != nil {
		fmt.Println("net Listen error")
		return
	}
	defer listener.Close()

	//4 启动服务
	grpcServer.Serve(listener)
}

客户端:

go 复制代码
package main

import (
	"context"
	"fmt"
	"google.golang.org/grpc"
	"google.golang.org/grpc/credentials/insecure"
	"log"
	"rpc_day01/pb"
)

func main() {
	//连接grpc服务(无加密)
	conn, err := grpc.Dial("127.0.0.1:8801", grpc.WithTransportCredentials(insecure.NewCredentials()))
	if err != nil {
		log.Fatal(err)
	}
	defer conn.Close()

	//初始化grpc客户端
	grpcClient := pb.NewHelloClient(conn)
	resp, err := grpcClient.SayHello(context.Background(), &pb.HelloRequest{RequestName: "李白"})
	if err != nil {
		log.Fatal(err)
	}
	fmt.Println(resp.ResponseMsg)
}
相关推荐
XMYX-014 小时前
37 - Go env 环境变量:配置管理与运行时控制
开发语言·golang
smileNicky16 小时前
Spring框架懒加载怎么实现?
python·spring·rpc
姚不倒20 小时前
Go 进阶实战:实现泛型数据验证器
云原生·golang
XMYX-021 小时前
36 - Go exec 执行命令
开发语言·golang
lolo大魔王21 小时前
Go 语言 HTTP 协议与 RESTful API 实训全解(理论 + 实战 + 规范)
http·golang·restful
一只小逸白1 天前
LeetCode Go 常用函数速查表
linux·leetcode·golang
郭老二1 天前
【C++】RPC:远程程序调用
c++·rpc
LCG元1 天前
【Go后端开发】从 0 到生产级:高性能分布式网关全实现 + 接口限流熔断降级实战
分布式·golang·wpf
姚不倒1 天前
Go语言进阶:接口、错误处理与并发编程(goroutine/channel/context)
云原生·golang
宇明一不急2 天前
go 链表 (标准库实现)
开发语言·链表·golang