23

golang中的net/rpc包

 5 years ago
source link: https://studygolang.com/articles/16603?amp%3Butm_medium=referral
Go to the source link to view the article. You can view the picture content, updated content and better typesetting reading experience. If the link is broken, please click the button below to view the snapshot at that time.

本文先介绍RPC,然后go原生对RPC的使用,之后是介绍go语言中有哪些RPC框架以及一些其他常见的框架,最后是探究go语言中rpc的源码。

(1)首先介绍下什么RPC?

(2)RPC可以做什么?

(3)RPC与REST风格的API有什么不同?

(4)go语言中使用RPC

(5)常见的RPC框架

(6)RPC源码探究

一、什么是RPC?

RPC是Remote Procedure Call,其中文意思就是 远程过程调用,可以理解成 一台主机上的进程调用另一台主机的进程服务,由一方为其它若干个主机提供服务。从表面上看非常类似于http API,RPC的目的可以屏蔽不同语言之间的关联,最大程度上进行解耦,调用方不需要知道服务方是用什么语言编写和其实现,只要知道服务方的RPC对外服务就行。其本质就是进程间的一种通信方式,可以是本机也可以是不同主机。

二、RPC可以做什么?

API、进程间通信,主要用于分布式应用间通信。

三、RPC与REST风格API有什么不同?

本质区别就是REST是使用http协议,相比RPC的实现协议传输会传更多的内容,但是两个可以做相同的事情。

四、go语言中使用RPC

RPC分服务提供和服务使用,也就是服务端和客户端,我们先来编写服务端:

服务端:

(1)服务内容

// 对外提供的必须是对外可见的类型
type Arith int

// 对外提供的方法也要是对外可见的类型,其中要被注册的服务至少要有一个对外可见的方法,不然执行的时候的时候会打印错误,还有对外服务的必须是方法第一个参数必须是对外可见的类型,第二个参数可以是对外可见类型或者是内置类型,然后必须要有一个返回值。
func (t *Arith) Multiply(args *Args, reply *int) error {
    *reply = args.A * args.B
    return nil
}

(2)端口监听

l, err := net.Listen("tcp", ":12345")

(3)注册服务

t := new(GetServerTime)
// 注册到RPC
err = rpc.Register(t)

(4)开启服务

rpc.HandleHTTP()

(5)启动HTTP服务

http.Serve(l, nil)

相比服务端,客户端的编写会简单很多,

客户端:

(1)连接服务RPC

client, err := rpc.DialHTTP(协议, ip:端口)

(2)调用RPC服务

有两种方式:同步或异步

// 同步方式
client.Call("rpc上的公开类名:公开方法", 第一个传入的变量, 第二个传入的变量)
// 异步方式
divCall := client.Go("rpc上的公开类名:公开方法", 第一次传入的变量, 第二个传入的变量, nil)
replyCall := <- divCall.Done        阻塞,等待异步完成

最基本的流程就是这样,然后下面一段例子,是从标准库那边copy过来的:

// 服务器端
package server

type Args struct {
    A, B int
}

type Quotient struct {
    Quo, Rem int
}

type Arith int
func (t *Arith) Multiply(args *Args, reply *int) error {
    *reply = args.A * args.B
    return nil
}

func (t *Arith) Divide(args *Args, quo *Quotient) error {
    if args.B == 0 {
        return errors.New("divide by zero")
    }
    quo.Quo = args.A / args.B
    quo.Rem = args.A % args.B
    return nil
}

func main() {
    arith := new(Arith)
    rpc.Register(arith)
    rpc.HandleHTTP()
    l, e := net.Listen("tcp", ":1234")
    if e != nil {
        log.Fatal("listen error:", e)
    }
    go http.Serve(l, nil)
}

// 客户端
package client

func main() {
      client, err := rpc.DialHTTP("tcp", serverAddress + ":1234")
      if err != nil {
    log.Fatal("dialing:", err)
}
然后,客户端可以执行远程调用:

// Synchronous call
args := &server.Args{7,8}
var reply int
err = client.Call("Arith.Multiply", args, &reply)
if err != nil {
    log.Fatal("arith error:", err)
}
fmt.Printf("Arith: %d*%d=%d", args.A, args.B, reply)
或:

// Asynchronous call
quotient := new(Quotient)
divCall := client.Go("Arith.Divide", args, quotient, nil)
replyCall := <-divCall.Done // will be equal to divCall
// check errors, print, etc.

五、常见的RPC框架

(1)Go:kiss-rpc、RPCX、grpc-go

(2)Java:Dubbo、HFS、Thrift

(3)php:yar

(4)python:Zerorpc

(5)多语言:Hession


About Joyk


Aggregate valuable and interesting links.
Joyk means Joy of geeK