go语言之路

2021你必须懂的RPC–分布式、微服务和云计算应用的核心技术

2021你必须懂的RPC--分布式、微服务和云计算应用的核心技术

随着分布式、微服务、云计算应用越来越火,对于程序猿来说,需要了解的东西也越来越多,概念也越来越多。今天就来聊聊这些热门技术背后的核心技术RPC--远程过程调用(Remote Procedure Call)。

http 与 rpc 的区别与联系

HTTP协议,以其中的Restful规范为代表,其优势很大。它可读性好,且可以得到防火墙的支持、跨语言的支持。但是HTTP也有其缺点,这是与其优点相对应的。首先是有用信息占比少,并且使用HTTP协议调用远程方法比较复杂,要封装各种参数名和参数值。

而良好的rpc调用是面向服务的封装,且RPC框架针对服务的可用性和效率等都做了优化。在目前推崇的分布式应用,微服务应用,云计算等面有着广泛的应用。可以理解为,远程的方法就和本地的方法无异,包括所有的过程都和本地是一样的。后面会使用代码进行讲解。

在分布式系统中,因为每个服务的边界都很小,很有可能调用别的服务提供的方法。这就出现了服务A调用服务B中方法的需求,即远程过程调用。要想让服务A调用服务B中的方法,最先想到的当然是通过http去实现,让服务B暴露接口,然后让A通过接口去调用对应的服务。

当然,基于接口的调用方式不仅可读性好,而且http请求可以通过各种防火墙,同时也很方便。但是基于这种方式存在一个明显的缺点,效率比较低,封装调用比较复杂,当前分布式,微服务比较热门,微服务就是将业务拆解为很多的小服务,那么这将存在大量的服务间调用,还使用基于接口的调用方式就不太适合了。

服务A调用服务B的过程是应用间的内部过程,牺牲可读性提升效率、易用性是可取的。基于这种思路,RPC产生了。调用方可以像调用内部接口一样调用远程的方法,而不用封装参数名和参数值等操作。

如何选择http 和 rpc呢?

根据业务场景,业务的大小进行灵活的运用。

与前端交互当然还是使用http的方式,在后端服务之间调用选择http 或者 rpc。

一般来讲中小厂多使用http方式。

大厂多使用http + rpc组合的方式。

通过代码分析两者的区别

我们实现一个简单的服务,两个整数相加,返回结果。

首先我们通过http方式实现。老样子使用Go语言做演示,技术栈为gin框架。

服务端代码:

package main

import (
    "github.com/gin-gonic/gin"
    "github.com/spf13/cast"
    "net/http"
)

func main() {
    r := gin.Default()
    //使用POST,路由为http://localhost:8080/add
    r.Handle("POST", "/add", add)
    if err := r.Run(":8080"); err != nil {
        panic("err")
    }
}

func add(c *gin.Context) {
    //获取参数a的值
    a := c.PostForm("a")
    //获取参数b的值
    b := c.PostForm("b")
    //将string类型转为int类型并相加
    result := cast.ToInt(a) + cast.ToInt(b)
    //返回json形式的结果,参数为result
    c.IndentedJSON(http.StatusOK, gin.H{
        "result" : result,
    })
}

image-20210303222351169

客户端代码:

package main

import (
    "fmt"
    "io/ioutil"
    "net/http"
    "net/url"
)

func main() {
    //将a = 1, b = 2传到后端
    params := url.Values{
        "a": {"1"},
        "b": {"2"},
    }
    //请求后端服务得到相应
    resp, _ := http.PostForm("http://localhost:8080/add", params)
    defer resp.Body.Close()
    //获取结果
    body, _ := ioutil.ReadAll(resp.Body)
    fmt.Println(string(body))
}

image-20210303222411773

接下来我们通过rpc方式实现

服务端代码:

package main

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

type NumbersTwo struct {
    A int
    B int
}

func main() {
    Nt := new(NumbersTwo)
    //注册服务对象
    //该对象必须是结构体对象,通过对象所属的方法暴露给调用者从而提供服务
    err := rpc.Register(Nt)
    if err != nil {
        log.Fatal(err)
    }
    //通过该函数把对象所属的方法的服务注册到http协议上,方便调用者可以利用http的方式进行数据传递
    rpc.HandleHTTP()
    //端口监听
    listen, err := net.Listen("tcp", ":8080")
    if err != nil {
        panic(err.Error())
    }
    err = http.Serve(listen, nil)
    if err != nil {
        panic(err.Error())
    }
}

func (n *NumbersTwo) AddTwoNumbers(t map[string]int, result *int) error {
    *result = t["A"] + t["B"]
    return nil
} 

客户端代码:

//client.go客户端
package main
import (
    "fmt"
    "net/rpc"
)
func main() {
    //通过http的端口监听方式连接
    client, err := rpc.DialHTTP("tcp", "localhost:8080")
    if err != nil {
        panic(err.Error())
    }
    var result *int
    t := map[string]int{
        "A":1,
        "B":2,
    }
    //调用远端方法
    err = client.Call("NumbersTwo.AddTwoNumbers", t, &result)
    if err != nil {
        panic(err.Error())
    }
    fmt.Println("远程调用程序, 结果为:", *result)
}

image-20210303223836889

使用rpc的方式,就像是使用本地的方法一样。比http方式更加的高效,更加的适合分布式应用和微服务。

web端数据实时更新是如何实现的?

上一篇

我竟然在路由器上跑了go程序--为什么不能在公共场合随便连接wifi!

下一篇

你也可能喜欢

发表评论

您的电子邮件地址不会被公开。 必填项已用 * 标注

提示:点击验证后方可评论!

插入图片

个人微信公众号

we-tuiguang

qq交流群

群号:1046260719

微信扫一扫

微信扫一扫