内容简介:本文先介绍RPC,然后go原生对RPC的使用,之后是介绍go语言中有哪些RPC框架以及一些其他常见的框架,最后是探究go语言中rpc的源码。(1)首先介绍下什么RPC?(2)RPC可以做什么?
本文先介绍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
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持 码农网
猜你喜欢:本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们。