转自: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. vue--拖动排序

    https://blog.csdn.net/jx950915/article/details/79803485?from=singlemessage

  2. Spark2 Dataset行列操作和执行计划

    Dataset是一个强类型的特定领域的对象,这种对象可以函数式或者关系操作并行地转换.每个Dataset也有一个被称为一个DataFrame的类型化视图,这种DataFrame是Row类型的Datas ...

  3. SqlServer数据库查询表信息/列信息(列ID/列名/数据类型/长度/精度/是否可以为null/默认值/是否自增/是否是主键/列描述)

    查询表信息(表名/表描述) Value ) AS value FROM sysobjects a Where a.xtype = 'U' AND a.name <> 'sysdiagram ...

  4. ubuntu16.04下安装Sophus

    git clone https://github.com/strasdat/Sophus.git 下载完成后 cd Sophus git checkout a621ffmkdir buildcd bu ...

  5. HDFS文件系统的JAVA-API操作(一)

    使用java.net.URL访问HDFS文件系统 HDFS的API使用说明: 1.如果要访问HDFS,HDFS客户端必须有一份HDFS的配置文件 也就是hdfs-site.xml,从而读取Nameno ...

  6. POJ 3522 - Slim Span - [kruskal求MST]

    题目链接:http://poj.org/problem?id=3522 Time Limit: 5000MS Memory Limit: 65536K Description Given an und ...

  7. POJ - 1101 The Game dfs

    题意:给你一个地图,上面有一些‘X',给你起点终点,让你输出从起点到终点的路径中转向(改变方向)次数最少的路径,注意,不能穿过别的’X'并且可以超过边界 题解:关于超过边界,只要在外围多加一圈‘ ’. ...

  8. CCCC L2-017. 人以群分 贪心

    https://www.patest.cn/contests/gplt/L2-017 题解:贪心,一点小数学 坑:XJB改下标改错了 #include <iostream> #includ ...

  9. 机器学习【算法】:KNN近邻

    引言 本文讨论的kNN算法是监督学习中分类方法的一种.所谓监督学习与非监督学习,是指训练数据是否有标注类别,若有则为监督学习,若否则为非监督学习.监督学习是根据输入数据(训练数据)学习一个模型,能对后 ...

  10. 走进APICloud的世界 (1)

    APICloud是什么东东?它是一个云端一体平台.啥意思?它利用HTML5跨平台技术同时满足android和ios的APP开发.相比APP传统开发而言,节约了不少成本,而且性能还可以和原生APP性能比 ...