使用 Spring 容器管理 Servlet
Servlet 可否也能像 Struts1/2 的 action 那样作为一个 javaBean 在 Spring 容器里进行管理呢?答案是肯定的。
自定义(继承自 javax.servlet.http.HttpServlet)的 Servlet 如何像 Struts1/2 中那样调用 Spring 容器的 service 呢?《Servlet 调用 Spring 容器的 service》一文很好地解决了这个问题。美中不足的是,ArcSyncDownloadServlet 在得到其注入的 bean 时,需要显式地写出 bean 在 Spring 配置中的 id 才可以:
this.setOperationService((OperationService) wac.getBean("operationService"));//Spring 配置 中的 bean id
这样子违背了 Spring 依赖注入的思想。那么如何才可以不在代码中显式调用这个 bean id,而把 bean id 直接写在配置文件中呢?
本文用一个项目中使用的例子介绍了将 Servlet 和业务对象的依赖关系使用 Spring 来管理,而不用再在 Servlet 中硬编码要引用对象的名字。
仍然使用《Servlet 调用 Spring 容器的 service》一文中的例子。
与《Servlet 调用 Spring 容器的 service》的做法相反,web.xml 和 Spring 的 application*.xml 配置需要改变,而 Servlet 不需要做改变。
如同 Struts1/2 的配置一样,Spring 在 web.xml 中的配置:
org.springframework.web.context.ContextLoaderListener
contextConfigLocation
/WEB-INF/applicationContext*.xml
如同 Struts1/2 的配置一样,Spring 在 applicationContext-service.xml 中定义我们的业务逻辑处理类:
如同一般的 Struts1/2 的 action 一样在我们的 Servlet 中注入 service:
private OperationService operationService = null;
public OperationService getOperationService() {
return operationService;
}
public void setOperationService(OperationService operationService) {
this.operationService = operationService;
}
在 Servlet 中如同一般的 Struts1/2 的 action 一样调用 service:
FileInfo fileInfo = this.getOperationService().getFileByFidAndSecret(Long.parseLong(fileId), secret);
如同一般的 Servlet 我们的这个 Servlet 需要继承 GenericServlet 或者 HttpServlet:
public class ArcSyncDownloadServlet extends HttpServlet
无须重写Servlet 的 init 方法,当然如果读者的业务需求需要的话除外。
新建一个 Servlet 代理类,这个类类似于 DelegatingFilterProxy 那样的代理,通过代理根据配置来找到实际的 Servlet,完成业务逻辑功能。
package com.defonds.cds.common;
import java.io.IOException;
import javax.servlet.Servlet;
import javax.servlet.ServletContext;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import org.springframework.web.context.WebApplicationContext;
import org.springframework.web.context.support.WebApplicationContextUtils;
public class DelegatingServletProxy extends HttpServlet {
private static final long serialVersionUID = 1L;
private String targetServletBean;
private Servlet proxy;
@Override
public void init() throws ServletException {
this.targetServletBean = this.getInitParameter("targetServletBean");
this.getServletBean();
this.proxy.init(this.getServletConfig());
}
@Override
protected void service(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
proxy.service(request,response);
}
private void getServletBean(){
ServletContext servletContext = this.getServletContext();
WebApplicationContext wac = null;
wac = WebApplicationContextUtils.getRequiredWebApplicationContext(servletContext);
this.proxy = (Servlet) wac.getBean(targetServletBean);
}
}
不同于一般的 Servlet 在 web.xml 中的配置,需要配置的是 Servlet 代理类,而非 Servlet:
proxyDownloadServletBean
com.defonds.cds.common.DelegatingServletProxy
targetServletBean
downLoadServletBean
proxyDownloadServletBean
/file
最后在 applicationContext-service.xml 中将 Servlet 及其业务对象的依赖关系配置到 Spring 容器管理:
注意这里的 downLoadServletBean 要和 web.xml 配置的 downLoadServletBean 对应,operationService 要和 Servlet 中 get/set 得到的 operationService bean 对应。
如果是多个 Servlet 的话,可以共用同一个代理 Servlet。Servlet 代码和代理 Servlet 代码无须改变,只需要注意一下在 web.xml 里的配置即可:
proxyDownloadServletBean
com.defonds.cds.service.operation.servlet.DelegatingServletProxy
targetServletBean
downLoadServletBean
proxyDownloadServletBean
/file
proxyWebServiceServletBean
com.defonds.cds.service.operation.servlet.DelegatingServletProxy
targetServletBean
webServiceServletBean
proxyWebServiceServletBean
/WebService