在讲之前,我们先来看看spring配置文件中为servlet注入到底能不能成功。首先,写一个随服务器启动的servlet,servlet中定义一个类变量,并定义set方法。随便在set方法中向控制台输出些东西,然后在doget,dopost或service方法中调用该类对象的一个方法。配置好配置文件,启动服务器。仔细观察控制台,我们发现servlet启动过程中已经执行了set方法,说明改servlet已经被成功注入。我们再打开浏览器访问下这个servlet,发现服务器报了空指针异常。大体意思是说你在doget或dopost或service方法中调用的那个变量是空。为什么会这样?其实也不难理解,我们使用spring的目的就是为了让spring为我们来提供一个已经被注入好的一个实例。而servlet是不同的,servlet是有生命周期的,而这个并不归属spring管理,而是由web容器管理的。那么当servlet刚刚创建的时候,spring可以为servlet注入,当你访问的时候,由于servlet是单实例多线程的,所以,servlet信息被重置,刚刚被注入的对象又为null了。 那么该怎么处理这个问题呢?其实也不难,只要我在获得serlet的时候,用从spring获得,而不是由web容器获得就可以了。要想获得spring中管理的bean肯定要获得applicationcontext对象,前面说过,在web开发中要获得applicationcontext对象需要获得servletcontext,所以,需要有一个servlet,而这个servlet所要做的就是获得spring中定义的那个servlet。而servlet最终还是要归web容器管理的,所以要归还给web容器,简单的讲就是在定义的这个servlet中所有方法都用从spring定义中的获得的那个servlet去处理就可以了。所以,这个servlet我们可以写成:public class servlettobeanproxy extends genericservlet { private string targetbean; private servlet proxy; public void init() throws servletexception { system.out.println("proxy init"); this.targetbean = getinitparameter("targetbean"); getservletbean(); proxy.init(getservletconfig()); } public void service(servletrequest req, servletresponse res) throws servletexception, ioexception { proxy.service(req, res); } private void getservletbean() { // ---------- linstner版 ------------ // webapplicationcontext wac = // webapplicationcontextutils.getrequiredwebapplicationcontext(getservletcontext()); // this.proxy = (servlet) wac.getbean(targetbean); //------------ servlet版 ------------ applicationcontext context = webapplicationcontextutils .getrequiredwebapplicationcontext(this.getservletcontext()); this.proxy = (servlet) context.getbean(targetbean); //通过servletcontext获得 // applicationcontext context = (applicationcontext) this // .getservletcontext() // .getattribute(webapplicationcontext.root_web_application_context_attribute); } } web.xml可以做如下配置:proxybean com.up72.servlet.servlettobeanproxy targetbeanactionservlet1proxybean*.do其中init-param中的targetbean就是你在spring中配置的那个servlet。 其他的j2ee api的注入网上都有配置方法,有兴趣的可以去查找。我就不列举了。之所以讲servlet是因为在mvc中servlet充当了控制器的角色,是mvc的关键和核心。在servlet往往需要大量注入service层的一些类,然后在类中调用。虽然,配置好listener或是servlet以后,你可以通过servletcontext的getattribute获得application对象,然后调用getbean()方法获得你所需要的bean,不过,这样就写死了。违背了软件开发的可修改原则。 备注:如果你只需要spring的注入功能,那么你只需要两个jar包就可以了。spring.jar和commons-logging.jar
spring管理filter和servlet
在使用spring容器的web应用中,业务对象间的依赖关系都可以用context.xml文件来配置,并且由spring容器来负责依赖对象 的创建。如果要在filter或者servlet中使用spring容器管理业务对象,通常需要使用
webapplicationcontextutils.getrequiredwebapplicationcontext(getservletcontext())来获得webapplicationcontext,然后调用webapplicationcontext.getbean("beanname")来获得对象的引用,这实际上是使用了依赖查找来获得对象,并且在filter或者servlet代码中硬编码了应用对象的bean名字。为了能在filter或者servlet中感知spring中bean,可采用如下步骤来实现:
1- 将filter或者servlet作为bean定义在context.xml文件中,和要应用的bean定义放在一起;
2- 实现一个filter代理或者servlet代理,该代理用webapplicationcontext来获得在context.xml中定义的filter或者servlet的对象,并将任务委托给context.xml中定义的filter或者servlet
3- 在web.xml中用contextloaderlistener来初始化spring 的context,同时在filter代理或者servlet代理的定义中用初始化参数来定义context.xml中filter或者servlet的bean名字(或者直接受用代理的名称获得相应的filter或者servlet的名称)。
4- 在web.xml中定义filter代理或者servlet代理的mapping.
利用这种方式就将filter或者servlet和业务对象的依赖关系用spring 来进行管理,并且不用在servlet中硬编码要引用的对象名字。
具体实例如下:
filter
1. 在applicationcontext.xml中定义filter
<bean id="springfilter" class="com.netqin.filter.springfilter">
<property name="name">
<value>springfilter</value>
</property>
</bean>
说明:com.netqin.filter.springfilter为实现了javax.servlet.filter接口的filter
2. 实现filter代理
实际上,filter代理不需要我们自己来实现,spring提供了两种现成的filter代理
org.springframework.security.util.filtertobeanproxy,
org.springframework.web.filter.delegatingfilterproxy,两者只是在web.xml中的配置上略有不同,下面就让我们一起看看如何在web.xml中进行配置。
3. 配置web.xml
ø 初始化spring的context
因为是使用spring来管理,所以在使用filter前先要初始化spring的context,一般来说配置如下:
<context-param>
<param-name>contextconfiglocation</param-name>
<param-value>
/web-inf/applicationcontext.xml
</param-value>
</context-param>
<listener>
<listener-class>
org.springframework.web.context.contextloaderlistener
</listener-class>
</listener>
ø filter配置:
² filtertobeanproxy
<filter>
<filter-name> springfilter </filter-name>
<filter-class>
org.springframework.security.util.filtertobeanproxy
</filter-class>
<init-param>
<param-name>targetbean</param-name>
<param-value>springfilter</param-value>
</init-param>
</filter>
说明:需要为filtertobeanproxy提供上下文参数,这里我们配置的是targetbean属性,它告诉spring在context中查找的bean名称,所以当请求被过滤器拦截后filtertobeanproxy会在applicationcontext.xml中会查找id为springfilter的bean.
我们也可以配置targetclass属性,意思就是查找该类型的bean.
² delegatingfilterproxy
<filter>
<filter-name>springfilter</filter-name>
<filter-class>
org.springframework.web.filter.delegatingfilterproxy
</filter-class>
</filter>
说明:使用delegatingfilterproxy时不需要配置任何参数,spring会根据filter-name的名字来查找bean,所以这里spring会查找id为springfilter的bean.
4. 配置filter的mapping
<filter-mapping>
<filter-name>springfilter</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>
ok!filter配置完成。推荐使用delegatingfilterproxy,应为配置上更简单。
servlet
servlet的配置与filter的配置十分相似
1. 在applicationcontext.xml中定义servlet
<bean id="springservlet" class="com.netqin.servlet.springservlet">
<property name="name">
<value>springservlet</value>
</property>
</bean>
说明:com.netqin.servlet.springservlet继承自
javax.servlet.http.httpservlet
2. 实现servlet代理
与filter不同,spring没有为servlet提供代理实现,需要我们自己来创建,不过放心,创建一个servlet代理十分简单,一个具体的实现如下:
import java.io.ioexception;
import javax.servlet.genericservlet;
import javax.servlet.servlet;
import javax.servlet.servletexception;
import javax.servlet.servletrequest;
import javax.servlet.servletresponse;
import org.springframework.web.context.webapplicationcontext;
import org.springframework.web.context.support.webapplicationcontextutils;
public class servlettobeanproxy extends genericservlet {
private string targetbean;
private servlet proxy;
public void init() throws servletexception {
this.targetbean = getinitparameter("targetbean");
getservletbean();
proxy.init(getservletconfig());
}
public void service(servletrequest req, servletresponse res)
throws servletexception, ioexception {
proxy.service(req, res);
}
private void getservletbean() {
webapplicationcontext wac = webapplicationcontextutils
.getrequiredwebapplicationcontext(getservletcontext());
this.proxy = (servlet) wac.getbean(targetbean);
}
}
说明:相信看了代码就明白了,它利用targetbean属性在spring中查找相应的servlet,
这很像filtertobeanproxy的方式,所以我为其取名为servlettobeanproxy。当然,我们也可以使用类似于delegatingfilterproxy的方式,只需要将上述代码中标记为黄色的部分修改为this.targetbean =this.getservletname();即可,我们相应的命名为delegatingservletproxy。
3. 配置web.xml
ø 初始化spring的context
与filter中的说明一致,不再赘述。
ø servlet配置:
² servlettobeanproxy
<servlet>
<servlet-name>springservlet</servlet-name>
<servlet-class>
com.netqin.servlet.proxy.servlettobeanproxy
</servlet-class>
<init-param>
<param-name>targetbean</param-name>
<param-value>springservlet</param-value>
</init-param>
<load-on-startup>1</load-on-startup>
</servlet>
² delegatingservletproxy
<servlet>
<servlet-name>springservlet</servlet-name>
<servlet-class>
com.netqin.servlet.proxy.delegatingservletproxy
</servlet-class>
<load-on-startup>1</load-on-startup>
</servlet>
4. 配置servlet的mapping
<filter-mapping>
<filter-name>springservlet</filter-name>
<url-pattern>/servlet/*</url-pattern>
</filter-mapping>
ok!servlet的配置完成。推荐使用delegatingservletproxy,应为配置上更简单。
Spring配Servlet
猜你喜欢
转载自haouziwefe.iteye.com/blog/1207274
今日推荐
周排行