如何讓servelt和Filter注入到spring容器或者在servlet中呼叫spring中的bean
在應用中一般普通的JavaPojo都是由Spring來管理的,所以使用autowire註解來進行注入不會產生問題,但是有兩個東西是例外的,一個是 Filter,一個是Servlet,這兩樣東西都是由Servlet容器來維護管理的,所以如果想和其他的Bean一樣使用Autowire來注入的 話,是需要做一些額外的功夫的。
對於Filter,Spring提供了DelegatingFilterProxy,所以本文主要講述Servlet的解決。
1、比較直觀但是不大優雅的做法是重寫init()方法獲取ServletContext 內建物件在整合到spring容器中去,具體寫法(兩種情況註解和非註解):
使用註解:
private ServletContext servletContext;
public void init(ServletConfig config) throws ServletException {
servletContext=config.getServletContext();
}
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
ServletContext application = (ServletContext) servletContext;
WebApplicationContext wac = WebApplicationContextUtils.getWebApplicationContext(application);// 獲取spring的context
//獲取userService的bean
UserService clientDao = (UserService) wac.getBean("userService");
int userId=Integer.parseInt(request.getParameter("userId"));
Tuser user=clientDao.getSingleUser(userId).get(0);
}
使用註解:
public void init(ServletConfig servletConfig) throws ServletException {
ServletContext servletContext = servletConfig.getServletContext();
WebApplicationContext webApplicationContext = WebApplicationContextUtils.getWebApplicationContext(servletContext);
AutowireCapableBeanFactory autowireCapableBeanFactory = webApplicationContext.getAutowireCapableBeanFactory();
autowireCapableBeanFactory.configureBean(this, BEAN_NAME);
}
其中,BEAN_NAME就是需要注入的Bean在spring中註冊的名字.
這樣寫的主要問題是就是那個BEAN_NAME,這樣寫有點主動查詢,而不是依賴注入的感覺。
第二種方式:
建立一個類似於DelegatingFilterProxy那樣的代理,通過代理根據配置來找到實際的Servlet,完成業務邏輯功能。
假定我們有一個Servlet名字叫UserServlet,需要注入一個UserManager,虛擬碼如下:
public class UserServlet extends HttpServlet {
@Autowired(required = true)
private UserManager userManager;
}
第一步:
public class DelegatingServletProxy extends GenericServlet {
private String targetBean;
private Servlet proxy;
@Override
public void service(ServletRequest req, ServletResponse res) throws ServletException, IOException {
proxy.service(req, res);
}
@Override
public void init() throws ServletException {
this.targetBean = getServletName();
getServletBean();
proxy.init(getServletConfig());
}
private void getServletBean() {
WebApplicationContext wac = WebApplicationContextUtils.getRequiredWebApplicationContext(getServletContext());
this.proxy = (Servlet) wac.getBean(targetBean);
}
}
第二步:
配置web.xml檔案,原來UserServlet的配置大致是這樣的:
<servlet>
<servlet-name>userServlet</servlet-name>
<servlet-class>com.sample.UserServlet</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>userServlet</servlet-name>
<url-pattern>/userServlet</url-pattern>
</servlet-mapping>
現在修改為:
<servlet>
<servlet-name>userServlet</servlet-name>
<servlet-class>com.sample.DelegatingServletProxy</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>userServlet</servlet-name>
<url-pattern>/userServlet</url-pattern>
</servlet-mapping>
或者另一種:
<servlet-name>menueInitServlet</servlet-name>
<servlet-class>cc.jqkj.oa.servlet.DelegatingServletProxy</servlet-class>
<init-param>
<param-name>menue-init-file</param-name>
<param-value>WEB-INF/menue.properties</param-value>
</init-param>
<load-on-startup>2</load-on-startup>
</servlet>
注意,spring是根據Servlet的名字來查詢被代理的Servlet的,所以,首先我們要在UserServlet類前面加上 @Component,來告訴Srping:我也是一個Bean。如果名稱和Web.xml裡面定義的不一樣的話,可以在這裡指定Bean的名字,比如: @Component(“userServlet”)
spring配置 檔案中: