使用servlet实现远程调用

 让我们先从这个servlet开始:

public class RemoteServlet extends HttpServlet {
 public void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException ,java.io.IOException{
  System.out.print("BaseDataServlet service received a remote request");
  ObjectInputStream in = new ObjectInputStream(req.getInputStream());
        resp.setContentType("application/octest-stream");
        ByteArrayOutputStream byteout = new ByteArrayOutputStream();
        ObjectOutputStream out = new ObjectOutputStream(byteout);
        Request request;
  try {
   request = (Request)in.readObject();//读取远程调用请求的封装对象
   //System.out.println(":/n"+request.toString());
   RequestProcessor processor=new RequestProcessor();//请求解析对象
         out.writeObject(processor.processorLocalhost(request));//执行请求并回写结果
         out.flush();
  } catch (ClassNotFoundException e) {   
   e.printStackTrace();
  }       
        byte buf[]= byteout.toByteArray();
        resp.setContentLength(buf.length);
        ServletOutputStream servletout = resp.getOutputStream();
        servletout.write(buf);
        servletout.close(); 
 }

这个servle从输入流读取远程调用的请求(Request对象):request = (Request)in.readObject();再由解析器RequestProcessor调用请求的对象,然后将结果回传给请求客户端。Request封装了请求的信息:

/**
 * 对远程调用请求的封装
 * @author wuxinyang
 */
public class Request implements java.io.Serializable{
 //目标地址,如果为localhost则为本地方法调用。
 private String server;
 //spring bean的配置名称,如果配置了该值则通过spring配置装载service。
 private String serviceBeanName;
 //服务的接口class名称,必须带包名
 private String serviceInterface;
 //要调用的方法名称
 private String methodName;
 //返回类型,该值不是必须的。
 private String returnType="void";
 //传入的参数列表
 private java.util.List arguments=new java.util.ArrayList();
 /**
  * 构造方法
  * @param server 目标地址
  * @param serviceInterface spring bean的配置名称
  * @param methodName 方法名称
  */
 public Request(String server, String serviceInterface, String methodName) {
  super();
  this.server = server;
  this.serviceInterface = serviceInterface;
  this.methodName = methodName;
 }
 /**
  * 添加调用参数
  * @param arg 调用参数
  */
 public void addArgument(Object arg){
  arguments.add(arg);
 }
 public java.util.List getArguments() {
  return arguments;
 }
 public void setArguments(java.util.List arguments) {
  this.arguments = arguments;
 }
 public String getMethodName() {
  return methodName;
 }
 public void setMethodName(String methodName) {
  this.methodName = methodName;
 }
 public String getReturnType() {
  return returnType;
 }
 public void setReturnType(String returnType) {
  this.returnType = returnType;
 }
 public String getServer() {
  return server;
 }
 public void setServer(String server) {
  this.server = server;
 }
 public String getServiceInterface() {
  return serviceInterface;
 }
 public void setServiceInterface(String serviceInterface) {
  this.serviceInterface = serviceInterface;
 }
 public String getServiceBeanName() {
  return serviceBeanName;
 }
 public void setServiceBeanName(String serviceBeanName) {
  this.serviceBeanName = serviceBeanName;
 }
 /**
  * @see java.lang.Object#toString()
  */
 public String toString() {
  return new ToStringBuilder(this).append("serviceInterface",
    this.serviceInterface).append("serviceBeanName",
    this.serviceBeanName).append("returnType", this.returnType)
    .append("arguments", this.arguments).append("server",
      this.server).append("methodName", this.methodName)
    .toString();
 } 
}
RequestProcessor类有如下方法:

public Object processorLocalhost(Request request) {
  String serviceInterface = request.getServiceInterface();//服务的接口class名称(被调用者的接口)
  try {
   Object service;
   if(request.getServiceBeanName()!=null){
    //通过Spring初始化
    service=com.westerasoft.common.SpringBeanFactory.getBean(request.getServiceBeanName());
   }
   else{
    //初始化
    service = Class.forName(serviceInterface).newInstance();
   }
   Method method = null;
   BeanInfo beanInfo = Introspector.getBeanInfo(service.getClass());
   MethodDescriptor[] methodDescriptors = beanInfo.getMethodDescriptors();
   for (int i = 0; i < methodDescriptors.length; i++) {
    MethodDescriptor descriptor = methodDescriptors[i];
    method = descriptor.getMethod();
    if (method.getName().equalsIgnoreCase(request.getMethodName()))
     break;
    method = null;
   }
   if (method != null) {
    //调用请求的方法
    return method.invoke(service, request.getArguments().toArray());
   }
  } catch (Exception e) {
   e.printStackTrace();
  }
  return null;
 }

客户端类如下:

public class RequestProcessorTest {
 public static void main(String[] args) {
  String urlString = "http://localhost:8080/wcommons/servlet/BaseDataServlet";
  //urlString="localhost";
  //请求调用远程接口com.westerasoft.common.dnet.RemoteService的方法gbToUtf8(String src)
  Request request = new Request(urlString,
    "com.westerasoft.common.dnet.RemoteService", "gbToUtf81");
  request.addArgument("中国人民解放军ABC");//添加调用参数即方法gbToUtf81的参数  
   Object answer = (Object) processor(request );
  System.out.println(answer);
  request = new Request(urlString,null,"findUsers");
  //请求调用远程接口userManager(Spring配置的bean name)的findUsers()方法
  request.setServiceBeanName("userManager");    answer = (Object) processor(request );
  System.out.println(answer);
 }

private Object processor(Request request) {
  try {
   URL url = new URL(request.getServer());
   java.net.URLConnection con = url.openConnection();
   con.setUseCaches(true);
   con.setDoOutput(true);
   con.setDoInput(true);
   con.setRequestProperty("Content-type", "application/octest-stream");
   con.setRequestProperty("Content-length", "" + -1);
   ObjectOutputStream dataout = new ObjectOutputStream(con
     .getOutputStream());
   dataout.writeObject(request);
   dataout.flush();
   dataout.close();
   ObjectInputStream in = new ObjectInputStream(con.getInputStream());
   Object obj = in.readObject();
   in.close();
   return obj;
  } catch (Exception e) {
   e.printStackTrace();
  }
  return null;
 } 
}

是不是极其简单(^_^),可以实现远程接口调用。记住你是通过servlet服务提供的,所以你也可以很容易的控制只对经过授权的用户能够远程调用。实际应用中你也许需要更完善和强大的远程调用框架,比如支持hibernate的延迟加载特性,和异构系统等等,那么有很多好的解决方案供你选择,已知的EJB,Spring,Webservice等等,都是你可以考虑的好东东。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值