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,应为配置上更简单。 

猜你喜欢

转载自haouziwefe.iteye.com/blog/1207274