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)
}
相关推荐
jerry6094 小时前
7天用Go从零实现分布式缓存GeeCache(改进)(未完待续)
分布式·缓存·golang
杜杜的man4 小时前
【go从零单排】Closing Channels通道关闭、Range over Channels
开发语言·后端·golang
问道飞鱼7 小时前
【微服务知识】开源RPC框架Dubbo入门介绍
微服务·rpc·开源·dubbo
极地星光8 小时前
JSON-RPC-CXX深度解析:C++中的远程调用利器
c++·rpc·json
甘橘籽9 小时前
【RPC】 gRPC、pb基本使用--经验与总结
golang
杜杜的man10 小时前
【go从零单排】HTTP客户端和服务端
开发语言·http·golang
材料苦逼不会梦到计算机白富美10 小时前
golang分布式缓存项目 Day6 防止缓存击穿
分布式·缓存·golang
杜杜的man13 小时前
【go从零单排】Environment Variables环境变量
golang
向阳121814 小时前
Dubbo使用Nacos作为注册中心
java·rpc·dubbo
材料苦逼不会梦到计算机白富美15 小时前
golang HTTP基础
http·golang·iphone