转自:http://blog.csdn.net/rulon147/article/details/53814589

一、什么是RPC

RPC(Remote Procedure Call Protocol)——远程过程调用协议,它是一种通过网络从远程计算机程序上请求服务,而不需要了解底层网络技术的协议。RPC协议假定某些传输协议的存在,如TCP或UDP,为通信程序之间携带信息数据。在OSI网络通信模型中,RPC跨越了传输层应用层。RPC使得开发包括网络分布式多程序在内的应用程序更加容易。
RPC采用客户机/服务器模式。请求程序就是一个客户机,而服务提供程序就是一个服务器。首先,客户机调用进程发送一个有进程参数的调用信息到服务进程,然后等待应答信息。在服务器端,进程保持睡眠状态直到调用信息到达为止。当一个调用信息到达,服务器获得进程参数,计算结果,发送答复信息,然后等待下一个调用信息,最后,客户端调用进程接收答复信息,获得进程结果,然后调用执行继续进行。
有多种 RPC模式和执行。最初由 Sun 公司提出。IETF ONC 宪章重新修订了 Sun 版本,使得 ONC RPC 协议成为 IETF 标准协议。现在使用最普遍的模式和执行是开放式软件基础的分布式计算环境(DCE)。

二、图例说明

三、java 实例演示

 

1、实现技术方案

下面使用比较原始的方案实现RPC框架,采用Socket通信、动态代理与反射与Java原生的序列化。

2、RPC框架架构

RPC架构分为三部分:

  1. 服务提供者,运行在服务器端,提供服务接口定义与服务实现类。
  2. 服务中心,运行在服务器端,负责将本地服务发布成远程服务,管理远程服务,提供给服务消费者使用。
  3. 服务消费者,运行在客户端,通过远程代理对象调用远程服务。

3、 具体实现

1)服务提供者接口定义与实现,代码如下:

  1. package services;
  2. public interface HelloService {
  3. String sayHi(String name);
  4. }

2)HelloServices接口实现类:

  1. package services.impl;
  2. import services.HelloService;
  3. public class HelloServiceImpl implements HelloService {
  4. public String sayHi(String name) {
  5. return "Hi, " + name;
  6. }
  7. }

3)服务中心代码实现,代码如下:

  1. package services;
  2. import java.io.IOException;
  3. public interface Server {
  4. public void stop();
  5. public void start() throws IOException;
  6. public void register(Class serviceInterface, Class impl);
  7. public boolean isRunning();
  8. public int getPort();
  9. }

4)服务中心实现类:

  1. package services.impl;
  2. import java.io.IOException;
  3. import java.io.ObjectInputStream;
  4. import java.io.ObjectOutputStream;
  5. import java.lang.reflect.Method;
  6. import java.net.InetSocketAddress;
  7. import java.net.ServerSocket;
  8. import java.net.Socket;
  9. import java.util.HashMap;
  10. import java.util.concurrent.ExecutorService;
  11. import java.util.concurrent.Executors;
  12. import services.Server;
  13. public class ServiceCenter implements Server {
  14. private static ExecutorService              executor        = Executors.newFixedThreadPool(Runtime.getRuntime()
  15. .availableProcessors());
  16. private static final HashMap<String, Class> serviceRegistry = new HashMap<String, Class>();
  17. private static boolean                      isRunning       = false;
  18. private static int                          port;
  19. public ServiceCenter(int port) {
  20. this.port = port;
  21. }
  22. public void stop() {
  23. isRunning = false;
  24. executor.shutdown();
  25. }
  26. public void start() throws IOException {
  27. ServerSocket server = new ServerSocket();
  28. server.bind(new InetSocketAddress(port));
  29. System.out.println("start server");
  30. try {
  31. while (true) {
  32. // 1.监听客户端的TCP连接,接到TCP连接后将其封装成task,由线程池执行
  33. executor.execute(new ServiceTask(server.accept()));
  34. }
  35. } finally {
  36. server.close();
  37. }
  38. }
  39. public void register(Class serviceInterface, Class impl) {
  40. serviceRegistry.put(serviceInterface.getName(), impl);
  41. }
  42. public boolean isRunning() {
  43. return isRunning;
  44. }
  45. public int getPort() {
  46. return port;
  47. }
  48. private static class ServiceTask implements Runnable {
  49. Socket clent = null;
  50. public ServiceTask(Socket client) {
  51. this.clent = client;
  52. }
  53. public void run() {
  54. ObjectInputStream input = null;
  55. ObjectOutputStream output = null;
  56. try {
  57. // 2.将客户端发送的码流反序列化成对象,反射调用服务实现者,获取执行结果
  58. input = new ObjectInputStream(clent.getInputStream());
  59. String serviceName = input.readUTF();
  60. String methodName = input.readUTF();
  61. Class<?>[] parameterTypes = (Class<?>[]) input.readObject();
  62. Object[] arguments = (Object[]) input.readObject();
  63. Class serviceClass = serviceRegistry.get(serviceName);
  64. if (serviceClass == null) {
  65. throw new ClassNotFoundException(serviceName + " not found");
  66. }
  67. Method method = serviceClass.getMethod(methodName, parameterTypes);
  68. Object result = method.invoke(serviceClass.newInstance(), arguments);
  69. // 3.将执行结果反序列化,通过socket发送给客户端
  70. output = new ObjectOutputStream(clent.getOutputStream());
  71. output.writeObject(result);
  72. } catch (Exception e) {
  73. e.printStackTrace();
  74. } finally {
  75. if (output != null) {
  76. try {
  77. output.close();
  78. } catch (IOException e) {
  79. e.printStackTrace();
  80. }
  81. }
  82. if (input != null) {
  83. try {
  84. input.close();
  85. } catch (IOException e) {
  86. e.printStackTrace();
  87. }
  88. }
  89. if (clent != null) {
  90. try {
  91. clent.close();
  92. } catch (IOException e) {
  93. e.printStackTrace();
  94. }
  95. }
  96. }
  97. }
  98. }
  99. }

5)客户端的远程代理对象:

  1. package client;
  2. import java.io.ObjectInputStream;
  3. import java.io.ObjectOutputStream;
  4. import java.lang.reflect.InvocationHandler;
  5. import java.lang.reflect.Proxy;
  6. import java.net.InetSocketAddress;
  7. import java.net.Socket;
  8. import java.lang.reflect.Method;
  9. public class RPCClient<T> {
  10. @SuppressWarnings("unchecked")
  11. public static <T> T getRemoteProxyObj(final Class<?> serviceInterface, final InetSocketAddress addr) {
  12. // 1.将本地的接口调用转换成JDK的动态代理,在动态代理中实现接口的远程调用
  13. return (T) Proxy.newProxyInstance(serviceInterface.getClassLoader(), new Class<?>[] { serviceInterface },
  14. new InvocationHandler() {
  15. public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
  16. Socket socket = null;
  17. ObjectOutputStream output = null;
  18. ObjectInputStream input = null;
  19. try {
  20. // 2.创建Socket客户端,根据指定地址连接远程服务提供者
  21. socket = new Socket();
  22. socket.connect(addr);
  23. // 3.将远程服务调用所需的接口类、方法名、参数列表等编码后发送给服务提供者
  24. output = new ObjectOutputStream(socket.getOutputStream());
  25. output.writeUTF(serviceInterface.getName());
  26. output.writeUTF(method.getName());
  27. output.writeObject(method.getParameterTypes());
  28. output.writeObject(args);
  29. // 4.同步阻塞等待服务器返回应答,获取应答后返回
  30. input = new ObjectInputStream(socket.getInputStream());
  31. return input.readObject();
  32. } finally {
  33. if (socket != null)
  34. socket.close();
  35. if (output != null)
  36. output.close();
  37. if (input != null)
  38. input.close();
  39. }
  40. }
  41. });
  42. }
  43. }

6)最后为测试类:

  1. package client;
  2. import java.io.IOException;
  3. import java.net.InetSocketAddress;
  4. import services.HelloService;
  5. import services.Server;
  6. import services.impl.HelloServiceImpl;
  7. import services.impl.ServiceCenter;
  8. public class RPCTest {
  9. public static void main(String[] args) throws IOException {
  10. new Thread(new Runnable() {
  11. public void run() {
  12. try {
  13. Server serviceServer = new ServiceCenter(8088);
  14. serviceServer.register(HelloService.class, HelloServiceImpl.class);
  15. serviceServer.start();
  16. } catch (IOException e) {
  17. e.printStackTrace();
  18. }
  19. }
  20. }).start();
  21. HelloService service = RPCClient
  22. .getRemoteProxyObj(HelloService.class, new InetSocketAddress("localhost", 8088));
  23. System.out.println(service.sayHi("test"));
  24. }
  25. }
 

运行结果:

  1. regeist service HelloService
  2. start server
  3. Hi, test

RPC框架原理剖析(含实例)(转)的更多相关文章

  1. 一个高性能RPC框架原理剖析

    业务与底层网络通信分离 Server大部分主要分为两层: 网络接收层:负责监听端口,负责收包,编码,解码工作,负责将响应包回传给客户端. 业务处理层:负责接收网络接收层完整的包,如果是RPCserve ...

  2. JQuery数组详解(含实例)

    <!doctype html>jQuery数组处理详解(含实例演示)@Mr.Think 演示所用数组 var _mozi=['墨家','墨子','墨翟','兼爱非攻','尚同尚贤']; 1 ...

  3. Spring中@Transactional事务回滚(含实例详细讲解,附源码)

    一.使用场景举例 在了解@Transactional怎么用之前我们必须要先知道@Transactional有什么用.下面举个栗子:比如一个部门里面有很多成员,这两者分别保存在部门表和成员表里面,在删除 ...

  4. 【转】python文件和目录操作方法大全(含实例)

    python文件和目录操作方法大全(含实例) 这篇文章主要介绍了python文件和目录的操作方法,简明总结了文件和目录操作中常用的模块.方法,并列举了一个综合实例,需要的朋友可以参考下一.python ...

  5. mybaits入门(含实例教程和源码) http://blog.csdn.net/u013142781/article/details/50388204

    前言:mybatis是一个非常优秀的存储过程和高级映射的优秀持久层框架.大大简化了,数据库操作中的常用操作.下面将介绍mybatis的一些概念和在eclipse上的实际项目搭建使用. 一.mybati ...

  6. Spring中@Transactional事务回滚(含实例具体解说,附源代码)

    一.使用场景举例 在了解@Transactional怎么用之前我们必须要先知道@Transactional有什么用. 以下举个栗子:比方一个部门里面有非常多成员,这两者分别保存在部门表和成员表里面,在 ...

  7. spark 源码分析之十二 -- Spark内置RPC机制剖析之八Spark RPC总结

    在spark 源码分析之五 -- Spark内置RPC机制剖析之一创建NettyRpcEnv中,剖析了NettyRpcEnv的创建过程. Dispatcher.NettyStreamManager.T ...

  8. 我的书籍《深入解析Java编译器:源码剖析与实例详解》就要出版了

    一个十足的技术迷,2013年毕业,做过ERP.游戏.计算广告,在大公司呆过,但终究不满足仅对技术的应用,在2018年末离开了公司,全职写了一本书<深入解析Java编译器:源码剖析与实例详解> ...

  9. Mahout机器学习平台之聚类算法具体剖析(含实例分析)

    第一部分: 学习Mahout必需要知道的资料查找技能: 学会查官方帮助文档: 解压用于安装文件(mahout-distribution-0.6.tar.gz),找到例如以下位置.我将该文件解压到win ...

随机推荐

  1. Spring Framework核心概念之Bean生命周期管理

    目录 Spring Bean的生命周期 相关接口的分类 测试SpringBean生命周期的Demo程序 小结 Spring Bean的生命周期 Spring容器既Application或者WebApp ...

  2. HDU 6312 - Game - [博弈][杭电2018多校赛2]

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=6312 Problem Description Alice and Bob are playing a ...

  3. hadoop第一部分-安装、测试

    一.hadoop安装(本地模式及伪分布式安装) hadoop历史版本下载网站:http://archive.apache.org/dist/运行模式:    本地模式    yarn模式 hadoop ...

  4. Photoshop制作倒影的两种方法

    图片加了倒影,画面立刻变得生动起来.而用PS,制作倒影是如此的方便. 素材1 将素材1导入文档,ctrl+J复制图层,编辑-变换-垂直翻转将翻转的图层拖至下方 为翻转的图层添加图层蒙版,选中渐变工具, ...

  5. Zero-Copy技术

    概述 考虑这样一种常用的情形:你需要将静态内容(类似图片.文件)展示给用户.那么这个情形就意味着你需要先将静态内容从磁盘中拷贝出来放到一个内存buf中,然后将这个buf通过socket传输给用户,进而 ...

  6. query:callback

    function getName(callback) { setTimeout(function() { callback('Aaron') }, 1000) } //等待callback回调 get ...

  7. 【剑指offer】斐波那契数列

    一.题目: 大家都知道斐波那契数列,现在要求输入一个整数n,请你输出斐波那契数列的第n项.n<=39 二.思路: 式子: n=0时,f=0:n=1或者n=2时f=1:否则f=f(n-1)+f(n ...

  8. golang 的精髓--pipeline流水线,对现实世界的完美模拟

    https://blog.golang.org/pipelines https://www.cnblogs.com/junneyang/p/6215785.html 简介 Go语言的并发原语允许开发者 ...

  9. iOS9 & iOS10 & iOS11 HTTP 不能正常使用的解决办法

    iOS9 & iOS10 & iOS11 HTTP 不能正常使用的解决办法 xcode  ios 291.4k 次阅读  ·  读完需要 8 分钟 54 今天升级Xcode 7.0 b ...

  10. 软件包管理:源码包管理-源码包与RPM包的区别

    RPM会安装在默认位置,源码包会安装在指定位置. 绝大多数ROM包遵循此规范.写软件包是就固定好了. 主要影响体现在对服务的管理方法,服务的管理分三种:启动,重启动,停止. 启动时使用绝对路径来启动. ...