基于Spring的RPC通讯模型.

一、概念和原理

    RPC(remote procedure call),远程过程调用,是客户端应用和服务端之间的会话。在客户端,它所需要的一些功能并不在该应用的实现范围之内,所以应用要向提供这些功能的其他系统寻求帮助。而远程应用通过远程服务暴露这些功能。RPC 是同步操作,会阻塞调用代码的执行,直到被调用的过程执行完毕。

    Spring支持多种不同的RPC模型,包括RMI、Caucho的Hessian和Burlap以及Spring自带的HTTP invoker:

    客户端:

    在所有的模型中,服务都是作为 Spring 所管理的 bean 配置到我们的应用中。这是通过一个代理工厂 bean 实现的,这个bean能够把远程服务像本地对象一样装配到其他bean的属性中。

    客户端向代理发起调用,就像代理提供了这些服务一样。代理代表客户端和远程服务进行通信,由它负责处理连接的细节并向远程服务发起调用。

    服务端:

Spring 使用远程导出器(remote exporter)将bean方法发布为远程服务。

回到顶部

二、RMI

    RMI 最初在JDK 1.1被引入到Java平台中,它为Java开发者提供了一种强大的方法来实现Java程序间的交互。

    Spring 提供了简单的方式来发布RMI服务,在服务端,RmiServiceExporter 可以把任何 Spring 管理的bean发布为RMI服务 ,如图所示,RmiServiceExporter 把bean包装在一个适配器类中,然后适配器类被绑定到RMI注册表中,并且代理到服务类的请求。 

/**

     * 服务端:

     * <p>

     * 1、默认情况下,RmiServiceExporter 会尝试绑定到本地机器1099端口上的RMI注册表。

     * 2、如果在这个端口没有发现RMI注册表,RmiServiceExporter 将会启动一个注册表。

     * 3、可重写注册表的路径和端口,这个是个大坑,当你设置了registryHost属性的时候,源码中就不创建Registry,而是直接去获取,可是我们自己也没有创建,所以就会报连接不上。

     *

     * @param userService

     * @return

     */

    @Bean(name = "rmiServiceExporter")

    public RmiExporter rmiServiceExporter(UserService userService, Environment environment) {

        String registryHost = environment.getProperty("registryHost");

        int registryPort = environment.getProperty("registryPort", Integer.class);

        RmiExporter rmiExporter = new RmiExporter();

        rmiExporter.setService(userService); //要把该bean(即rmiServiceImpl)发布为一个RMI服务

        rmiExporter.setServiceName("RmiService"); //命名RMI 服务

        rmiExporter.setServiceInterface(UserService.class); //指定服务所实现的接口

        rmiExporter.setRegistryHost(registryHost);

        rmiExporter.setRegistryPort(registryPort);

        return rmiExporter;

    }

/**

 * Created by XiuYin.Cui on 2018/5/14.

 * 

 * 解决设置 registryHost 后,报连接拒绝的问题。

 */

public class RmiExporter extends RmiServiceExporter {


    @Override

    protected Registry getRegistry(String registryHost, int registryPort, RMIClientSocketFactory clientSocketFactory,

                                   RMIServerSocketFactory serverSocketFactory) throws RemoteException {



        if (registryHost != null) {

            try {

                if (logger.isInfoEnabled()) {

                    logger.info("Looking for RMI registry at port '" + registryPort + "' of host [" + registryHost + "]");

                }

                //把spring源代码中这里try起来,报异常就创建一个

                Registry reg = LocateRegistry.getRegistry(registryHost, registryPort, clientSocketFactory);

                testRegistry(reg);

                return reg;

            } catch (RemoteException ex) {

                LocateRegistry.createRegistry(registryPort);

                Registry reg = LocateRegistry.getRegistry(registryHost, registryPort, clientSocketFactory);

                testRegistry(reg);

                return reg;

            }

        } else {

            return getRegistry(registryPort, clientSocketFactory, serverSocketFactory);

        }

    }

}

接下来,来看看客户端是怎么使用这些远程服务的吧!Spring的RmiProxyFactoryBean是一个工厂bean,该bean可以为RMI服务创建代理。该代理代表客户端来负责与远程的RMI服务进行通信。客户端通过服务的接口与代理进行交互,就如同远程服务就是一个本地的POJO。

@Resource(name="rmiUserServiceClient")    private UserService userService;

RMI 的缺陷:

1、RMI很难穿越防火墙,这是因为RMI使用任意端口来交互——这是防火墙通常所不允许的。
2、RMI是基于Java的。这意味着客户端和服务端必须都是用java开发。因为RMI使用了Java的序列化机制,所以通过网络传输的对象类型必须要保证在调用两端的Java运行时中是完全相同的版本。

    tips:最近发现 Dubbo 底层也是用 RMI 实现的,它把 zookeeper 当作注册表。

回到顶部

三、Hessian 和 Burlap

    Hession 和 Burlap 是 Caucho Technology 的两种基于HTTP的轻量级远程服务解决方案。借助于尽可能简单的API和通信协议,它们都致力于简化Web服务。

    Hession,像RMI一样,使用二进制消息进行客户端和服务端的交互。但是它与RMI不同的是,它的二进制消息可以移植到其他非Java的语言中。由于它是基于二进制的,所以它在带宽上更具优势。

    Burlap 是一种基于XML的远程调用技术,这使得它可以自然而然的移植到任何能够解析XML的语言上。正因为它基于XML,所以相比起Hessian的二进制格式而言,Burlap可读性更强。但是和其他基于XML的远程技术(例如SOAP或XML-RPC)不同,Burlap的消息结构尽可能的简单。

    下面我们会介绍 Hession 的使用。Spring 不推荐使用 Burlap,BurlapServiceExporter 在4.0后被废弃,不再提供支持。5.0 后直接从开发包丢弃了。

    服务端,类似于 RmiServiceExporter ,Hession 也有一个 HessianServiceExporter 将 Spring 管理的 bean 发布为 Hessian 服务,不同于RMI的是,HessianServiceExporter是一个Spring MVC控制器,它接收Hessian请求(HTTP协议的请求),并将这些请求转换成对被导出POJO的方法调用。既然是HTTP请求,那我们就必须配置Spring 的 DispatcherServlet ,并配置 HandlerMapping,将相应的URL映射给 HessianServiceExporter。

 

/**

     * Hession没有注册表,不需要设置 serviceName

     */

    @Bean(name = "hessianServiceExporter")

    public HessianServiceExporter hessianServiceExporter(UserService userService) {

        HessianServiceExporter hessianServiceExporter = new HessianServiceExporter();

        hessianServiceExporter.setService(userService);

        hessianServiceExporter.setServiceInterface(UserService.class);

        return hessianServiceExporter;

    }

    /**

     * 需要配置一个URL映射来确保DispatcherServlet把请求转给HessianServiceExporter

     */

    @Bean(name = "handlerMapping")

    public HandlerMapping handlerMapping() {

        SimpleUrlHandlerMapping handlerMapping = new SimpleUrlHandlerMapping();

        Properties mappings = new Properties();

        mappings.setProperty("/user.service", "hessianServiceExporter");

        handlerMapping.setMappings(mappings);

        return handlerMapping;

    }

客户端,类似于 RmiProxyFactoryBean ,Hessian 也有一个代理工厂Bean——HessianProxyFactoryBean,来创建代理与远程服务进行通信:


esource(name="hessianUserServiceClient")    private UserService userService; 

Hessian 的缺陷:

    Hession 和 Burlap 都是基于HTTP的,它们都解决了RMI所头疼的防火墙渗透问题。但是当传递过来的RPC消息中包含序列化对象时,RMI就完胜 Hessian 和 Burlap 了。因为 Hessian 和 Burlap 都采用了私有的序列化机制,而RMI使用的是Java本身的序列化机制。

回到顶部

四、HttpInvoker

    RMI 和 Hessian 各有自己的缺陷,一方面,RMI使用Java标准的对象序列化机制,但是很难穿透防火墙。另一方面,Hessian和Burlap能很好地穿透防火墙,但是使用私有的对象序列化机制。就这样,Spring的HTTP invoker应运而生了。HTTP invoker是一个新的远程调用模型,作为Spring框架的一部分,能够执行基于HTTP的远程调用,并使用Java的序列化机制。

    HttpInvoker 的使用和 Hessian 很类似,HttpInvokerServiceExporter 也是一个Spring MVC 控制器,也是通过 DispatcherServlet 将请求分发给它...

客户端,像 RmiProxyFactoryBean 和 HessianProxyFactoryBean 一样,HttpInvoker 也提供了一个代理工厂Bean——HttpInvokerProxyFactoryBean,用于创建HttpInvoker代理来与远程服务通信:

@Resource(name="httpInvokerUserServiceClient")    private UserService userService;


©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,772评论 6 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,458评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,610评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,640评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,657评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,590评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,962评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,631评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,870评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,611评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,704评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,386评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,969评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,944评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,179评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,742评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,440评论 2 342

推荐阅读更多精彩内容