Servlet的学习之ServletContext(转)

Servlet的学习之ServletContext

Servlet 的三种创建方式http://www.runoob.com/note/27872

Tomcat源码分析:web.xml解析https://sq.163yun.com/blog/article/189808829158150144

我们在浏览器点击链接和按钮产生的消息不是发送给Servlet的,而是发送给web容器的(在JSP出现之前,web容器也叫Servlet容器),web容器接收消息后不知道怎么处理,转交给我们编写的Servlet处理,那么web容器怎么和Servlet交流呢?于是就出现了Servlet接口,接口是定义一种规范的良好表达形式。只要我们编写的Java类符合Servlet规范,那么就能被Web容器识别并被容器管理。
当servlet配置了初始化参数后,web容器在创建servlet实例对象时,会自动将这些初始化参数封装到ServletConfig对象中,并在调用servlet的init方法时,将ServletConfig对象传递给servlet。进而,程序员通过ServletConfig对象就可以得到当前servlet的初始化参数信息。
本篇来介绍Servlet中非常重要的对象,如ServletConfig类和ServletContext类,尤其是ServletContext类中的一些方法。

在对Servlet配置的web.xml文件中,经常会使用一些初始化的参数来配置Servlet,总的功能来说就是不在Servlet程序中将某个变量写死,而是通过外界(如web.xml文件)进行传递,同时便于修改。这个是使用标签下的标签,使用标签的来封装一个键值对,可以使用多个标签进行多个初始化参数的设定,我们可以看看Tomcat的web.xml中的默认Servlet:

image

可以看到在这个默认Servlet中有两个初始化参数,分别是“debug=0”和“listings=false”。

当Servlet在web.xml文件中配置了标签后,web容器会在创建Servlet实例对象时,自动将这些初始化参数封装到ServletConfig对象中,并在调用Servlet的初始化init方法时,将ServletConfig对象传递给Servlet。

我们从Servlet接口的初始化方法:init(ServletConfig config),可以知道,当服务器创建Servlet对象就将ServletConfig对象传递,而在ServletConfig对象中包含着标签所配置的参数和值。

刚开始学Servlet时,就已经谈到过Servlet接口的非生命周期方法就有一个方法是getServletConfig()方法,返回ServletConfig对象。所以当我们在开发的Servlet的web.xml文件中配置一些信息:

image

而在Servlet中的程序获取这个配置的参数:

public void doGet(HttpServletRequest request, HttpServletResponse response)
            throws ServletException, IOException {

        ServletConfig config = this.getServletConfig();
        String initValue = config.getInitParameter("love");
        System.out.println(initValue);
    }

重新部署该web应用,然后在浏览器来访问这个Servlet,将会看到在MyEclipse的控制台上打印出:

image

在ServletConfig类中,getInitParameter(String name)方法是传入特定参数名来获取对应参数的值,getInitParameterNames()方法则是将所有的参数名装进一个Enumeration对象返回,当我们有多个参数键值对时:

image

在Servlet中进行遍历和输出:

public void doGet(HttpServletRequest request, HttpServletResponse response)
            throws ServletException, IOException {

        ServletConfig config = this.getServletConfig();
        Enumeration initParams = config.getInitParameterNames();
        while(initParams.hasMoreElements()) {
            String paramName = (String)initParams.nextElement();
            String paramValue = config.getInitParameter(paramName);
            System.out.println(paramName+" = "+paramValue );
        }
    }
image

最后,ServletConfig对象的作用通常用于获得编码表类型,获得数据库连接信息,获得配置文件(如Struts的web.xml文件中)等等。

说完了ServletConfig对象,当我们去看这个对象的方法时会发现这个方法中还有一个方法getServletContext()是返回一个ServletContext对象,这是Servlet中一个非常重要的类。当然ServletContext对象还可以从父类的方法中直接获取。

Web容器在启动时会为每个web应用创建一个ServletContext对象,而这个ServletContext对象就代表当前这个web应用。因为一个ServletContext对象代表一个web应用,所以该web应用中所有的Servlet和其他资源都共享一个ServletContext对象,这时,我们就可以通过ServletContext对象进行Servlet对象之间的通讯。而ServletContext对象也称之为Context域对象。

我们先来看看ServletContext对象的获取的两种方式:

public void doGet(HttpServletRequest request, HttpServletResponse response)
            throws ServletException, IOException {

        //两种获取ServletContext对象的方法:
        ServletContext context1 = this.getServletConfig().getServletContext();
        ServletContext context2 = this.getServletContext();
        
        //System.out.println(context1 == context2);   //ture
    }

可以通过先获取ServletConfig对象来获取,或者直接通过父类的方法来获取,这两种方式获取到的是同一对象(相同地址)。

既然说ServletContext代表这个web应用,我们可以用它来进行Servlet直接的通讯,那么我们就创建一个工程来进行两个Servlet之间的数据传输。在一个【myservlet】web工程下创建两个Servlet:ServletDemo1和ServletDemo2,

ServletDemo1在ServletContext中设置参数键值对,代码为:

public void doGet(HttpServletRequest request, HttpServletResponse response)
            throws ServletException, IOException {

        ServletContext context = this.getServletContext();
        context.setAttribute("lover", "LRR");
    }

ServletDemo2从ServletContext中获取键值对,代码为:

public void doGet(HttpServletRequest request, HttpServletResponse response)
            throws ServletException, IOException {

        ServletContext context = this.getServletContext();    
        System.out.println(context.getAttribute("lover"));
    }

在浏览器先访问ServletDemo1后(先执行ServletDemo1才能使ServletContext设置参数),再访问ServletDemo2后,MyEclipse的控制台就输出了ServletContext中设置的参数,这就达到了从一个Servlet传递数据给另一个Servlet。当然这只是ServletContext的一个小小应用。

在ServletContext类中还有getInitParameter(String name)方法或者getInitParameterNames()方法,这两个方法获取的是web应用所配置的参数(毕竟ServletContext代表web应用),就像ServletConfig中类似的方法获取的是某个Servlet中的标签配置的参数。

而对于配置web应用的参数是在web.xml文件中使用标签,正如在该文件中为Servlet配置参数而使用标签一样。这种配置标签的好处在于属于全局性的配置,而每个Servlet的配置参数仅局限于在Servlet的范围内,举个例子,对于整个web应用配置数据库连接,这样在web应用中的每个Servlet都可以使用,而无需再在每个Servlet中都单独设置一次,提高了效率。

例:在【myservlet】web工程下建立了名为ServletDemo3的Servlet,并在该web工程下的web.xml文件中添加标签作为该web应用的配置参数:

image

在ServletDemo3中的代码如下:

public void doGet(HttpServletRequest request, HttpServletResponse response)
            throws ServletException, IOException {
        
        ServletContext context = this.getServletContext();
        String username = context.getInitParameter("username");
        String password = context.getInitParameter("password");
        
        System.out.println(username +":"+ password);
}

在浏览器中访问该Servlet,如果MyEclipse的控制台能打印该信息,说明每个Servlet可以通过ServletContext对象来获取web应用的配置信息,也从侧面说明了ServletContext代表了这个web应用:

image

ServletContext类中的getMimeType(String file)方法用于返回该文件的MIME类型:

public void doGet(HttpServletRequest request, HttpServletResponse response)
            throws ServletException, IOException {
        
        String filename = "1.html";
        ServletContext context = this.getServletContext();
        System.out.println(context.getMimeType(filename));    
    }

输出:text/html。

ServletContext中的转发方法(重要)

在ServletContext对象中还有这么两个方法:getNameDispatcher(String name)(不常用)和getRequestDispatcher(String path),返回的是RequestDispatcher对象。转发有什么作用呢,举个例子,比如一个Servlet中的数据交个另一个Servlet来处理,或者Servlet将某个实时数据交给JSP来显示,虽然我们在浏览器中访问的是最开始的Servlet,但是进行转发后看到的其他web资源,而浏览器的地址栏不会改变。

注:在请求对象request对象中也有这么一个getRequestDispatcher(String path)方法,功能与ServletContext对象的这个方法一样,也可以实现转发,因此用哪个对象都行,没有区别。

例:在【myservlet】web工程下创建一个名为ServletDemo1的Servlet和一个show.jsp,

在ServletDemo1中将数据转发给show.jsp,代码为:

public void doGet(HttpServletRequest request, HttpServletResponse response)
            throws ServletException, IOException {
    String data = "Ding love LRR";
    this.getServletContext().setAttribute("data", data);  //将数据存至web应用的配置中
    ServletContext context = this.getServletContext();        
    RequestDispatcher dispathcer = context.getRequestDispatcher("/show.jsp");  //通过要转发的目的地来获取转发对象
    dispathcer.forward(request, response);      //通过forward方法将请求对象和响应对象转发给别人处理
    }

而在show.jsp中接收这个数据,并封装在HTML中:


        ${data }

接着我们去浏览器里访问ServletDemo1,就会看到:

image

虽然我们请求的ServletDemo1资源,但是由于在ServletDemo1中将请求进行了转发,所以其实服务器返回的是show.jsp的资源,但是我们浏览器地址依然会是ServletDemo1,这也是转发和重定向的区别之一。

(补充:其实使用ServletContext对象将数据转发至JSP并不合理,可能在多线程中,一个Servlet利用ServletContext在转发给JSP过程中,而另一个线程中的Servlet使用ServletContext将这个转发的数据给覆盖,这样导致原先该转发给JSP的数据并不是我们期待的,所以使用ServletContext的getRequestDispatcher方法不如使用request请求对象的getRequestDispatcher适用于实际开发场景)。

Tomcat如何创建Servlet?

A.先到缓存中寻找有没有这个对象
如果没有: 1、通过反射去创建相应的对象(执行构造方法)
2、tomcat会把对象存放到缓存中
3、执行初始化方法init
如果有该对象,直接获取到这个对象
B. 执行服务方法
C.返回响应的数据到客户端(浏览器)

Servlet的执行流程

一:从浏览器地址(请求)开始分析
1.http://localhost:80/servlet/hello

  1. Localhost:80 -> 找到我们的服务器
  2. 到tomcat的server.xml中找到 Context 这个配置
  3. servlet: 这个Servlet找到Context中的path
  4. 通过path找到它对应的docBase,也就是找到我们工程实际地址
  5. Hello 就到我们实现的项目中找到相应的
    web.xml文件中的servlet-mapping中的url-pattern
image.png

image.png

你可能感兴趣的:(Servlet的学习之ServletContext(转))