Servlet的学习之ServletContext(1)
本篇来介绍Servlet中非常重要的对象,如ServletConfig类和ServletContext类,尤其是ServletContext类中的一些方法,本篇先讲述一部分,在下一篇中继续补充。
在对Servlet配置的web.xml文件中,经常会使用一些初始化的参数来配置Servlet,总的功能来说就是不在Servlet程序中将某个变量写死,而是通过外界(如web.xml文件)进行传递,同时便于修改。这个是使用<servlet>标签下的<init-param>标签,使用<init-param>标签的<param-name>和<param-value>来封装一个键值对,可以使用多个<init-param>标签进行多个初始化参数的设定,我们可以看看Tomcat的web.xml中的默认Servlet:
可以看到在这个默认Servlet中有两个初始化参数,分别是“debug=0”和“listings=false”。
当Servlet在web.xml文件中配置了<init-param>标签后,web容器会在创建Servlet实例对象时,自动将这些初始化参数封装到ServletConfig对象中,并在调用Servlet的初始化init方法时,将ServletConfig对象传递给Servlet。
我们从Servlet接口的初始化方法:init(ServletConfig config),可以知道,当服务器创建Servlet对象就将ServletConfig对象传递,而在ServletConfig对象中包含着<init-param>标签所配置的参数和值。
刚开始学Servlet时,就已经谈到过Servlet接口的非生命周期方法就有一个方法是getServletConfig()方法,返回ServletConfig对象。所以当我们在开发的Servlet的web.xml文件中配置一些信息:
而在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的控制台上打印出:
在ServletConfig类中,getInitParameter(String name)方法是传入特定参数名来获取对应参数的值,getInitParameterNames()方法则是将所有的参数名装进一个Enumeration对象返回,当我们有多个参数键值对时:
在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 ); } }
最后,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中的<init-param>标签配置的参数。
而对于配置web应用的参数是在web.xml文件中使用<context-param>标签,正如在该文件中为Servlet配置参数而使用<init-param>标签一样。这种配置<context-param>标签的好处在于属于全局性的配置,而每个Servlet的配置参数仅局限于在Servlet的范围内,举个例子,对于整个web应用配置数据库连接,这样在web应用中的每个Servlet都可以使用,而无需再在每个Servlet中都单独设置一次,提高了效率。
例:在【myservlet】web工程下建立了名为ServletDemo3的Servlet,并在该web工程下的web.xml文件中添加<context-param>标签作为该web应用的配置参数:
在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应用:
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中:
<font size="100px" color="red"> ${data } </font>
接着我们去浏览器里访问ServletDemo1,就会看到:
虽然我们请求的ServletDemo1资源,但是由于在ServletDemo1中将请求进行了转发,所以其实服务器返回的是show.jsp的资源,但是我们浏览器地址依然会是ServletDemo1,这也是转发和重定向的区别之一。
(补充:其实使用ServletContext对象将数据转发至JSP并不合理,可能在多线程中,一个Servlet利用ServletContext在转发给JSP过程中,而另一个线程中的Servlet使用ServletContext将这个转发的数据给覆盖,这样导致原先该转发给JSP的数据并不是我们期待的,所以使用ServletContext的getRequestDispatcher方法不如使用request请求对象的getRequestDispatcher适用于实际开发场景)。