一文搞懂RPC,So Easy!

开发 前端
RPC主要依赖于客户端与服务端建立socket链接;而HTTP REST实现通讯的代价比较高,这是RPC的一个优势体现。

什么是RPC

RPC的中文是“远程过程调用”,对应的英文全称是:Remote Procedure Call,可以简单理解为一个节点请求另一个节点提供的服务。

请先自行思考一下什么是“本地过程调用”,可以更好的理解“远程过程调用”。

知识点:RPC主要依赖于客户端与服务端建立socket链接;而HTTP REST实现通讯的代价比较高,这是RPC的一个优势体现。(gRPC使用http2.0)

为什么用RPC

就是因为无法在同一个进程内,或者无法在同一个服务器上通过本地调用的方式实现我们的需求。

HTTP能满足需求但是不够高效,所以我们需要使用RPC。

知乎大神的回答[1]

RPC的优势

  1. RPC能够跨多种开发工具和平台
  2. RPC能够跨语言调用
  3. RPC能够提高系统的可扩展性,解耦,提高复用
  4. RPC相较于HTTP 1.1,传输效率更高,性能消耗更小,自带负载均衡策略,自动实现服务治理

RPC和HTTP对比

  • RPC主要用于公司内部的服务调用,性能消耗低,传输效率高,服务治理方便。
  • HTTP主要用于对外的异构环境,浏览器接口调用,APP接口调用,第三方接口调用等。
  • RPC和HTTP的详细对别[2]可以看这篇文章,不作为本篇的重点

RPC的使用边界

通过和HTTP的对比,我们可以倒推出RPC的边界:对外的异构环境,浏览器接口调用,APP接口调用,第三方接口调用。

上述这些都不适合RPC,不知道RPC不适合做什么,比知道RPC能做什么更重要。

RPC入门1:net/rpc

基本构成

  1. RPC的基本构成:服务端,客户端
  2. 服务端基本构成:结构体,请求结构体,响应结构体
  3. 客户端基本构成:请求结构体,响应结构体

代码示例

rpc_service.go

package main

import (
"errors"
"fmt"
"log"
"net"
"net/http"
"net/rpc"
"os"
)

type Arith struct {

}

//请求结构体
type ArithRequest struct {
A int
B int
}

//响应结构体
type ArithResponse struct {
Pro int //乘积
Quo int //
Rem int //余数
}

//乘积方法
func (this *Arith) Multiply(req ArithRequest,res *ArithResponse) error{
res.Pro = req.A * req.B
return nil
}

//除法运算方法
func (this *Arith) Divide(req ArithRequest,res *ArithResponse) error{
if req.B ==0 {
return errors.New("divide by zero")
}
res.Quo = req.A / req.B
res.Rem = req.A % req.B
return nil
}

func main() {
//注册rpc服务
rpc.Register(new(Arith))
//采用http协议作为rpc载体
rpc.HandleHTTP()

lis,err := net.Listen("tcp","127.0.0.1:8095")
if err!=nil {
log.Fatalln("fatal error:",err)
}

fmt.Fprintf(os.Stdout,"%s","start connection\n")

//常规启动http服务
http.Serve(lis,nil)
}

rpc_client.go

package main

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

//算数运算请求结构体
type ArithRequest struct {
A int
B int
}

//响应结构体
type ArithResponse struct {
Pro int //
Quo int //
Rem int //余数
}

func main() {
conn,err := rpc.DialHTTP("tcp","127.0.0.1:8095")
if err!=nil {
log.Fatalln("dialing error:",err)
}

req := ArithRequest{10,20}
var res ArithResponse

err = conn.Call("Arith.Multiply",req,&res) //乘法运算
if err!=nil {
log.Fatalln("arith error:",err)
}
fmt.Printf("%d * %d = %d\n",req.A,req.B,res.Pro)

//除法运算
err = conn.Call("Arith.Divide",req,&res)
if err!=nil {
log.Fatalln("arith error:",err)
}
fmt.Printf("%d / %d = %d 余数是:%d",req.A,req.B,res.Quo,res.Rem)
}

运行结果

先启动服务端,再启动客户端连接服务端

//服务端console
start connection

//客户端console
10 * 20 = 200
10 / 20 = 0 余数是:10

RPC入门2:net/rpc/jsonrpc

实现跨语言调用

jsonrpc_server.go

package main

import (
"errors"
"fmt"
"log"
"net"
"net/rpc"
"net/rpc/jsonrpc"
"os"
)

type Arith struct {

}

//请求结构体
type ArithRequest struct {
A int
B int
}

//响应结构体
type ArithResponse struct {
Pro int //乘积
Quo int //
Rem int //余数
}

//乘积方法
func (this *Arith) Multiply(req ArithRequest,res *ArithResponse) error{
res.Pro = req.A * req.B
return nil
}

//除法运算方法
func (this *Arith) Divide(req ArithRequest,res *ArithResponse) error{
if req.B ==0 {
return errors.New("divide by zero")
}
res.Quo = req.A / req.B
res.Rem = req.A % req.B
return nil
}

func main() {
//注册rpc服务
rpc.Register(new(Arith))
//采用http协议作为rpc载体
rpc.HandleHTTP()

lis,err := net.Listen("tcp","127.0.0.1:8096")
if err!=nil {
log.Fatalln("fatal error:",err)
}

fmt.Fprintf(os.Stdout,"%s","start connection\n")

//接收客户端请求 并发处理 jsonrpc
for {
conn,err :=lis.Accept() //接收客户端连接请求
if err!=nil {
continue
}

//并发处理客户端请求
go func(conn net.Conn) {
fmt.Fprintf(os.Stdout,"%s","new client in coming\n")
jsonrpc.ServeConn(conn)
}(conn)
}

//常规启动http服务
//http.Serve(lis,nil)
}

jsonrpc_client.go

package main

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

//算数运算请求结构体
type ArithRequest struct {
A int
B int
}

//响应结构体
type ArithResponse struct {
Pro int //
Quo int //
Rem int //余数
}

func main() {
// 只有这里不一样
conn,err := jsonrpc.Dial("tcp","127.0.0.1:8096")
if err!=nil {
log.Fatalln("dialing error:",err)
}

req := ArithRequest{9,2}
var res ArithResponse

err = conn.Call("Arith.Multiply",req,&res) //乘法运算
if err!=nil {
log.Fatalln("arith error:",err)
}
fmt.Printf("%d * %d = %d\n",req.A,req.B,res.Pro)

//除法运算
err = conn.Call("Arith.Divide",req,&res)
if err!=nil {
log.Fatalln("arith error:",err)
}
fmt.Printf("%d / %d = %d 余数是:%d",req.A,req.B,res.Quo,res.Rem)
}

运行结果

先启动服务端,再启动客户端连接服务端

//服务端console
start connection

//客户端console
9 * 2 = 18
9 / 2 = 4 余数是:1

//服务端console
new client in coming

RPC入门3:go php跨语言调用

Go作为服务端,PHP作为客户端

jsonrpc_server.go:和入门2服务端的代码一样

下面是PHP代码

jsonrpc_client.php

<?php
class JsonRPC
{

private $conn;

function __construct($host, $port)
{
$this->conn = fsockopen($host, $port, $errno, $errstr, 3);
if (!$this->conn) {
return false;
}
}

public function Call($method, $params)
{
if (!$this->conn) {
return false;
}
$err = fwrite($this->conn, json_encode(array(
'method' => $method,
'params' => array($params),
'id' => 0,
)) . "\n");
if ($err === false) {
return false;
}
stream_set_timeout($this->conn, 0, 3000);
$line = fgets($this->conn);
if ($line === false) {
return NULL;
}
return json_decode($line, true);
}
}

$client = new JsonRPC("127.0.0.1", 8096);
$args = array('A' => 9, 'B' => 2);
$r = $client->Call("Arith.Multiply", $args);
printf("%d * %d = %d\n", $args['A'], $args['B'], $r['result']['Pro']);
$r = $client->Call("Arith.Divide", array('A' => 9, 'B' => 2));
printf("%d / %d, Quo is %d, Rem is %d\n", $args['A'], $args['B'], $r['result']['Quo'], $r['result']['Rem']);

如何在本地启动PHP[3] 不作为本文重点,可以看这篇文章。

运行结果

本地启动PHP服务:

​http://127.0.0.1/jsonrpc_client.php​

运行结果如下:

9 * 2 = 18 9 / 2, Quo is 4, Rem is 1

总结

一文入门RPC,就是如此丝滑,So Easy!

欢迎还在用单体架构,没有使用RPC的同学们操练起来,尤其是PHP的小伙伴们,卷起来吧。该学学Go语言啦~

相关资料

[1]知乎大神的回答: https://www.zhihu.com/question/25536695

[2]RPC和HTTP的详细对别: http://www.ccutu.com/244407.html

[3]如何在本地启动PHP: ​​https://blog.csdn.net/resilient/article/details/80770531​

本文转载自微信公众号「 程序员升级打怪之旅」,作者「王中阳Go」,可以通过以下二维码关注。

转载本文请联系「 程序员升级打怪之旅」公众号。

责任编辑:武晓燕 来源: 程序员升职加薪之旅
相关推荐

2022-02-14 21:17:21

RPC框架协议

2024-04-12 12:19:08

语言模型AI

2022-03-24 08:51:48

Redis互联网NoSQL

2021-03-22 10:05:59

netstat命令Linux

2023-09-08 08:20:46

ThreadLoca多线程工具

2023-09-15 12:00:01

API应用程序接口

2022-08-15 15:39:23

JavaScript面向对象数据

2023-08-24 16:50:45

2021-01-13 05:21:59

参数

2021-06-30 08:45:02

内存管理面试

2022-06-07 10:13:22

前端沙箱对象

2020-03-18 14:00:47

MySQL分区数据库

2023-10-16 08:16:31

Bean接口类型

2019-11-19 08:00:00

神经网络AI人工智能

2022-05-30 10:37:35

分布式事务反向补偿

2021-03-04 00:09:31

MySQL体系架构

2023-03-06 21:29:41

mmap技术操作系统

2021-02-28 20:53:37

Cookie存储浏览器

2021-07-08 10:08:03

DvaJS前端Dva

2020-12-07 06:19:50

监控前端用户
点赞
收藏

51CTO技术栈公众号