java 反照机制(二)远程方法调用
java 反射机制(二)远程方法调用
假定在SimpleServer 服务器端创建了一个HelloServiceImpl对象,它具有getTime()
和echo()方法。HelloServiceImpl类实现了HelloService接口
远程方法调用先定义接口及其实现,
SimpleClient客户端如何调用服务器端的HelloServiceImpl对象的getTime()和echo()
方法呢?显然,SimpleClient客户端需要把调用的方法名、方法参数类型、方法参数值,
以及方法所属的类名或接口名发送给SimpleServer,SimpleServer 再调用相关对象的方
法,然后把方法的返回值发送给SimpleClient。
为了便于按照面向对象的方式来处理客户端与服务器端的通信,可以把它们发送
的信息用Call 类(如例程10-8所示)来表示。一个Call对象表示客户端发起的一个远
程调用,它包括调用的类名或接口名、方法名、方法参数类型、方法参数值和方法执
行结果。
例程10-8Call.java
SimpleClient 调用SimpleServer 端的HelloServiceImpl 对象的echo()方法的流程
如下。
(1)SimpleClient 创建一个Call 对象,它包含了调用HelloService 接口的echo()
方法的信息。
(2)SimpleClient通过对象输出流把Call对象发送给SimpleServer。
(3)SimpleServer 通过对象输入流读取Call 对象,运用反射机制调用
HelloServiceImpl对象的echo()方法,把echo()方法的执行结果保存到Call对象中。
(4)SimpleServer 通过对象输出流把包含了方法执行结果的Call 对象发送给
SimpleClient。
(5)SimpleClient通过对象输入流读取Call对象,从中获得方法执行结果。
如例程10-9 和例程10-10 所示分别是SimpleServer 和SimpleClient 的源程序。
客户端代码例程10-10 SimpleClient.java
服务器端代码:例程10-9 SimpleServer.java
先运行命令“java reflection.rmi.SimpleServer”,再运行命令“java reflection.rmi.SimpleClient”,SimpleClient 端将打印“echo:Hello”。该打印结果是服务器端执行HelloServiceImpl 对象的echo()方法的返回值
如下图所示:
假定在SimpleServer 服务器端创建了一个HelloServiceImpl对象,它具有getTime()
和echo()方法。HelloServiceImpl类实现了HelloService接口
远程方法调用先定义接口及其实现,
package reflection.rmi; import java.util.Date; public interface HelloService {//服务接口 public String echo(String msg); public Date getTime(); }
package reflection.rmi; import java.util.Date; public class HelloServiceImpl implements HelloService { //服务类 public String echo(String msg) { return "echo:" + msg; } public Date getTime() { return new Date(); } }
SimpleClient客户端如何调用服务器端的HelloServiceImpl对象的getTime()和echo()
方法呢?显然,SimpleClient客户端需要把调用的方法名、方法参数类型、方法参数值,
以及方法所属的类名或接口名发送给SimpleServer,SimpleServer 再调用相关对象的方
法,然后把方法的返回值发送给SimpleClient。
为了便于按照面向对象的方式来处理客户端与服务器端的通信,可以把它们发送
的信息用Call 类(如例程10-8所示)来表示。一个Call对象表示客户端发起的一个远
程调用,它包括调用的类名或接口名、方法名、方法参数类型、方法参数值和方法执
行结果。
例程10-8Call.java
package reflection.rmi; import java.io.*; public class Call implements Serializable { // 可串口化的对象 private String className; // 表示类名或接口名 private String methodName; // 表示方法名 private Class[] paramTypes; // 表示方法参数类型,用于getMethod方法 private Object[] params; // 表示方法参数值,用于invoke方法 // 表示方法的执行结果 // 如果方法正常执行,则result 为方法返回值,如果方法抛出异常,那么result 为该异常。 private Object result; public Call() { } public Call(String className, String methodName, Class[] paramTypes, Object[] params) { this.className = className; this.methodName = methodName; this.paramTypes = paramTypes; this.params = params; } public String getClassName() { return className; } public void setClassName(String className) { this.className = className; } public String getMethodName() { return methodName; } public void setMethodName(String methodName) { this.methodName = methodName; } public Class[] getParamTypes() { return paramTypes; } public void setParamTypes(Class[] paramTypes) { this.paramTypes = paramTypes; } public Object[] getParams() { return params; } public void setParams(Object[] params) { this.params = params; } public Object getResult() { return result; } public void setResult(Object result) { this.result = result; } public String toString() { return "className=" + className + " methodName=" + methodName; } }
SimpleClient 调用SimpleServer 端的HelloServiceImpl 对象的echo()方法的流程
如下。
(1)SimpleClient 创建一个Call 对象,它包含了调用HelloService 接口的echo()
方法的信息。
(2)SimpleClient通过对象输出流把Call对象发送给SimpleServer。
(3)SimpleServer 通过对象输入流读取Call 对象,运用反射机制调用
HelloServiceImpl对象的echo()方法,把echo()方法的执行结果保存到Call对象中。
(4)SimpleServer 通过对象输出流把包含了方法执行结果的Call 对象发送给
SimpleClient。
(5)SimpleClient通过对象输入流读取Call对象,从中获得方法执行结果。
如例程10-9 和例程10-10 所示分别是SimpleServer 和SimpleClient 的源程序。
客户端代码例程10-10 SimpleClient.java
package reflection.rmi; import java.io.*; import java.net.*; import java.util.*; public class SimpleClient { public void invoke() throws Exception { Socket socket = new Socket("localhost", 8888); OutputStream out = socket.getOutputStream(); ObjectOutputStream oos = new ObjectOutputStream(out); InputStream in = socket.getInputStream(); ObjectInputStream ois = new ObjectInputStream(in); // Call call=new Call("remotecall.HelloService","getTime", // new Class[]{},new Object[]{}); Call call = new Call("reflection.rmi.HelloService", "echo", new Class[] { String.class }, new Object[] { "Hello" }); oos.writeObject(call); // 向服务器发送Call 对象 call = (Call) ois.readObject(); // 接收包含了方法执行结果的Call 对象 System.out.println(call.getResult()); ois.close(); oos.close(); socket.close(); } public static void main(String args[]) throws Exception { new SimpleClient().invoke(); } }
服务器端代码:例程10-9 SimpleServer.java
package reflection.rmi; import java.io.*; import java.net.*; import java.util.*; import java.lang.reflect.*; public class SimpleServer { private Map remoteObjects = new HashMap(); // 存放远程对象的缓存 /** 把一个远程对象放到缓存中 */ public void register(String className, Object remoteObject) { remoteObjects.put(className, remoteObject); } public void service() throws Exception { ServerSocket serverSocket = new ServerSocket(8888); System.out.println("服务器启动......"); while (true) { Socket socket = serverSocket.accept(); InputStream in = socket.getInputStream(); ObjectInputStream ois = new ObjectInputStream(in); // 读取对象 OutputStream out = socket.getOutputStream(); ObjectOutputStream oos = new ObjectOutputStream(out); Call call = (Call) ois.readObject(); // 接收客户发送的Call 对象 System.out.println(call); // toString()方法 call = invoke(call); // 调用相关对象的方法 oos.writeObject(call); // 向客户发送包含了执行结果的Call 对象 ois.close(); oos.close(); socket.close(); } } public Call invoke(Call call) { Object result = null; try { String className = call.getClassName(); String methodName = call.getMethodName(); Object[] params = call.getParams(); Class classType = Class.forName(className); Class[] paramTypes = call.getParamTypes(); Method method = classType.getMethod(methodName, paramTypes); Object remoteObject = remoteObjects.get(className); // 从缓存中取出相关的远程对象 if (remoteObject == null) { throw new Exception(className + "的远程对象不存在"); } else { result = method.invoke(remoteObject, params); } } catch (Exception e) { result = e; } call.setResult(result); // 设置方法执行结果 return call; } public static void main(String args[]) throws Exception { SimpleServer server = new SimpleServer(); // 把事先创建的HelloServiceImpl 对象加入到服务器的缓存中 server.register("reflection.rmi.HelloService", new HelloServiceImpl()); server.service(); } }
先运行命令“java reflection.rmi.SimpleServer”,再运行命令“java reflection.rmi.SimpleClient”,SimpleClient 端将打印“echo:Hello”。该打印结果是服务器端执行HelloServiceImpl 对象的echo()方法的返回值
如下图所示: