Spring配Servlet

在讲之前,我们先来看看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,应为配置上更简单。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值