使用 Spring 容器管理 Servlet

使用 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
		

关于如何使用 Spring 管理 Filter,请参考作者的另一篇博客《 使用 Spring 容器管理 Filter》。

你可能感兴趣的:(使用 Spring 容器管理 Servlet)