我々は、リモートの方法でクライアントのサーバー側の実装を呼び出す方法を、ネットワークプログラミングでアプリケーションの反射を紹介します。
HelloServiceサーバインターフェイスを仮定すると、getTime()とエコー()メソッドを有するインターフェースは、特定のコードは次のようにされています。
import java.util.Date;
public interface HelloService {
public String echo(String msg);
public Date getTime();
}
サーバー上のクラスを作成し、HelloServiceインタフェースを実装HelloServiceImpl。HelloServiceImplクラスのコードは次のよう:
import java.util.Date;
public class HelloServiceImpl implements HelloService {
@Override
public String echo(String msg) {
return "echo:" + msg;
}
@Override
public Date getTime() {
return new Date();
}
}
HelloServiceImplエコー()メソッドとgetTime()メソッドのクラスに上記のコードに示すように書き直されています。だから、どのようにクライアントサーバハローServiceImplクラスgetTime()およびエコー()メソッドを呼び出すには?
具体的な方法はある:クラスまたはインタフェースの名前送るクライアントは、メソッド名、メソッドのパラメータの型、パラメータ値のメソッドを呼び出す必要があり、この方法は、サーバに属します。サーバー側の方法と、関連するオブジェクトを呼び出し、そのメソッドの戻り値は、クライアントに送信されます。
ハンドル通信クライアントとサーバに応じて、オブジェクト指向の方法を容易にするために、情報が表さコールクラスによってそれらに送信することができます。A Callオブジェクトは、クラスまたはインタフェース名、メソッド名、パラメータ型、メソッド、パラメータおよび実行結果のメソッドを呼び出す含むクライアントによって開始されたリモート呼び出しを表します。
次のようにコールクラスのコードは次のとおりです。
import java.io.Serializable;
public class Call implements Serializable {
private static final long serialVersionUID = 6659953547331194808L;
private String className; // 表示类名或接口名
private String methodName; // 表示方法名
private Class[] paramTypes; // 表示方法参数类型
private Object[] params; // 表示方法参数值
// 表示方法的执行结果
// 如果方法正常执行,则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は、以下の方法である)(エコーSimpleServerとにSimpleClientコールフローオブジェクト
1. SimpleClientは、Callオブジェクト情報を含む呼び出しHelloServiceインタフェースエコー()メソッドを作成します。
CallオブジェクトのSimpleServerとを送信するためにオブジェクトを介して2 SimpleClientストリーム出力。
結果は、エコー実行Callオブジェクトの()メソッドを保存するために、反射エコーHelloServiceImplオブジェクト呼び出し()メソッドを使用して、オブジェクトの入力ストリームを読み取ることによって3 SimpleServerとCallオブジェクト。
方法の結果Callオブジェクトを行う出力ストリーム含むを送信するオブジェクトを介して4 SimpleServerとSimpleClient。
入力ストリームオブジェクトを読み取ることによって前記SimpleClient Callオブジェクト、実行結果そこを得るための方法。
まず、コードを見てみましょう実装クライアントプログラムSimpleClientクラスという。
import java.io.*;
import java.net.*;
import java.util.*;
import java.lang.reflect.*;
import java.io.*;
import java.net.*;
import java.util.*;
public class SimpleClient {
public void invoke() throws Exception {
Socket socket = new Socket("localhost", 8000);
OutputStream out = socket.getOutputStream();
ObjectOutputStream oos = new ObjectOutputStream(out);
InputStream in = socket.getInputStream();
ObjectInputStream ois = new ObjectInputStream(in);
// 创建一个远程调用对象
Call call = new Call("ch12.HelloService", "echo", new Class[] { String.class }, new Object[] { "Java" });
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();
}
}
上記のコードに示すように、主な役割は、クライアントSimpleClientクラスは、サーバーとの接続を確立し、サーバーに送信される情報でCallオブジェクトを呼び出すことです。
呼要求を受信した後SimpleServerとクラスサーバは、動的に指定されたオブジェクトの指定されたメソッドを呼び出すためにリフレクションを使用して、クライアントに実行結果を返します。
SimpleServerとクラスのコードは次のとおりです。
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(8000);
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);
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("ch13.HelloService", new HelloServiceImpl());
server.service();
}
}
これは、サーバー側SimpleServerとを実行し、クライアントSimpleClientを実行するために、あなた最初の必要性、Webアプリケーションですので、。結果は結果がエコーサーバー側の実装HelloServicelmplオブジェクト()メソッドの戻り値である、「echoJava」の出力を参照するには、クライアント上で実行されています。次の図に示すSimpleServerとを有する通信処理SimpleClient。