rpc那点东西
2015-12-16 16:03
302 查看
rpc框架
最近学习了一些rpc框架,分享一下rpc原理。RpcFramework
package rpc; import java.io.ObjectInputStream; import java.io.ObjectOutputStream; import java.lang.reflect.InvocationHandler; import java.lang.reflect.Method; import java.lang.reflect.Proxy; import java.net.ServerSocket; import java.net.Socket; public class RpcFramework { /** * 暴露服务 * * @param service 服务实现 * @param port 服务端口 * @throws Exception */ public static void export(final Object service, int port) throws Exception { if (service == null) throw new IllegalArgumentException("service instance == null"); if (port <= 0 || port > 65535) throw new IllegalArgumentException("Invalid port " + port); System.out.println("Export service " + service.getClass().getName() + " on port " + port); ServerSocket server = new ServerSocket(port); for(;;) { try { final Socket socket = server.accept(); new Thread(new Runnable() { @Override public void run() { try { try { ObjectInputStream input = new ObjectInputStream(socket.getInputStream()); try { String methodName = input.readUTF(); Class<?>[] parameterTypes = (Class<?>[])input.readObject(); Object[] arguments = (Object[])input.readObject(); ObjectOutputStream output = new ObjectOutputStream(socket.getOutputStream()); try { Method method = service.getClass().getMethod(methodName, parameterTypes); Object result = method.invoke(service, arguments); output.writeObject(result); } catch (Throwable t) { output.writeObject(t); } finally { output.close(); } } finally { input.close(); } } finally { socket.close(); } } catch (Exception e) { e.printStackTrace(); } } }).start(); } catch (Exception e) { e.printStackTrace(); } } } /** * 引用服务 * * @param <T> 接口泛型 * @param interfaceClass 接口类型 * @param host 服务器主机名 * @param port 服务器端口 * @return 远程服务 * @throws Exception */ @SuppressWarnings("unchecked") public static <T> T refer(final Class<T> interfaceClass, final String host, final int port) throws Exception { if (interfaceClass == null) throw new IllegalArgumentException("Interface class == null"); if (! interfaceClass.isInterface()) throw new IllegalArgumentException("The " + interfaceClass.getName() + " must be interface class!"); if (host == null || host.length() == 0) throw new IllegalArgumentException("Host == null!"); if (port <= 0 || port > 65535) throw new IllegalArgumentException("Invalid port " + port); System.out.println("Get remote service " + interfaceClass.getName() + " from server " + host + ":" + port); return (T) Proxy.newProxyInstance(interfaceClass.getClassLoader(), new Class<?>[]{interfaceClass}, new InvocationHandler() { public Object invoke(Object proxy, Method method, Object[] arguments) throws Throwable { Socket socket = new Socket(host, port); try { ObjectOutputStream output = new ObjectOutputStream(socket.getOutputStream()); try { output.writeUTF(method.getName()); output.writeObject(method.getParameterTypes()); output.writeObject(arguments); ObjectInputStream input = new ObjectInputStream(socket.getInputStream()); try { Object result = input 4000 .readObject(); if (result instanceof Throwable) { throw (Throwable) result; } return result; } finally { input.close(); } } finally { output.close(); } } finally { socket.close(); } } }); } }
服务定义
package rpc; public interface HelloService { public String hello(); }
服务实现
package rpc; public class HelloServiceImpl implements HelloService { @Override public String hello() { System.out.println(this.getClass().getName() + " hello!"); return "rui.w"; } }
服务暴露
package rpc; public class Provider { public static void main(String[] args) throws Exception { HelloService service = new HelloServiceImpl(); RpcFramework.export(service,80); } }
引用服务
package rpc; public class Consumer { private static final String HOST = "127.0.0.1"; private static final int PORT = 80; public static void main(String[] args) throws Exception { HelloService helloService1 = RpcFramework.refer(HelloService.class, HOST, PORT); String hello = helloService1.hello(); System.out.println(hello); } }
相关文章推荐
- 深入浅出 RPC
- Hadoop的RPC通信(二)------>框架封装思想
- 浙大PAT 1049. 数列的片段和(20)
- LitePal 使用方法简单总结
- Android音频系统之AudioTrack(二)
- tableView contentInset的理解
- android.mk加入三方jar
- 深入理解MFC子类化
- Scala StatCounter类
- 查看ORACLE
- phpMyadmin安装极简教程[下载,解压,登录]
- Mongodb 聚合
- spring aop通配符的各种解释
- javascript:void(0)是什么意思
- 遍历map集合
- 第16周 项目2 - 大数据集上排序算法性能的体验
- Android音频系统之AudioTrack(一)
- <!doctype html>是什么意思?
- 每日一篇:将一个正整数分解质因数
- 【鸟瞰】C#的学习