初识ServeltContext

正文开始

在正式了解 ServletConfig 之前,先让大家有一个整体印象。

之前讲过ServletConfig, 我发现他们有两个共同的方法,我对他们的区别做了小小的总结

上面的思维导图是对下面内容的总结,大家可以先看看,对等会的内容有一个大概印象,看完后,可以再回头看看这个思维导图。

ServletContext是什么

  • ServletContext是一个接口,是Servlet规范中的一员,
  • ServletContext对象的创建也是Tomcat服务器来完成的。启动webapp的时候创建的。

ServletContext是谁实现的

java 复制代码
Tomcat服务器(WEB服务器)实现了ServletContext接口
public class org.apache.catalina.core.ApplicationContextFacade implements ServletContext{}
  • Servlet对象是谁创建的,在什么时候创建的

ServletContext对象是WEB服务器启动的时候创建的

ServletContext对象是WEB服务器创建的

ServletContext的生命周期

  • 对于一个webapp来说,ServletContext对象只有一个

  • 一个web应用对应一个ServletContext实例,这个实例是应用部署启动后,servlet容器为应用创建的。

    ServletContext对象在服务器关闭的时候销毁

  • 一个Servlet对象对应一个ServletConfig。100个Servlet对象则对应100个ServletConfig对象。

  • 只要在同一个webapp当中,只要在同一个应用当中,所有的Servlet对象都是共享同一个ServletContext对象的。

  • ServletContext对象在服务器启动阶段创建,在服务器关闭的时候销毁。这就是ServletContext对象的生命周期。ServletContext对象是应用级对象。

  • Tomcat服务器中有一个webapps,这个webapps下可以存放webapp,可以存放多个webapp,假设有100个webapp,那么就有100个ServletContext对象。但是,总之,一个应用,一个webapp肯定是只有一个ServletContext对象。

    接下来,通过程序验证一下

java 复制代码
        ServletContext servletContext = this.getServletContext();
        response.setContentType("text/html");
        PrintWriter writer = response.getWriter();
        writer.print("ServletContext对象:"+servletContext+"<br>");

一个webapp应用共享一个ServletContext对象

java 复制代码
        ServletContext servletContext = this.getServletContext();
        response.setContentType("text/html");
        PrintWriter writer = response.getWriter();
        writer.print("ServletContext对象:"+servletContext+"<br>");



在同一个webapp下的Servlet都是共享同一个ServletContext对象

  • ServletContext被称为Servlet上下文对象。(Servlet对象的四周环境对象。>)

  • 一个ServletContext对象通常对应的是一个web.xml文件。

  • ServletContext对应显示生活中的什么例子呢?

    • 一个教室里有多个学生,那么每一个学生就是一个Servlet,这些学生都在同一个教室当中,那么我们可以把这个教室叫做ServletContext对象。那么也就是说放在这个ServletContext对象(环境)当中的数据,在同一个教室当中,物品都是共享的。比如:教室中有一个空调,所有的学生都可以操作。可见,空调是共享的。因为空调放在教室当中。教室就是ServletContext对象。
    • Tomcat是一个容器,一个容器可以放多个webapp,一个webapp对应一个ServletContext对象

ServletContext接口中常用的方法

java 复制代码
public String getInitParameter(String name); // 通过初始化参数的name获取value
public Enumeration<String> getInitParameterNames(); // 获取所有的初始化参数的name

        ServletContext servletContext = this.getServletContext();
        response.setContentType("text/html");
        PrintWriter writer = response.getWriter();
        writer.print("ServletContext对象:"+servletContext+"<br>");
        String initParameter = servletContext.getInitParameter("pageSize");
        writer.println("pageSize=  "+initParameter);

  Enumeration<String> names = servletContext.getInitParameterNames();
        while (names.hasMoreElements()){

            String s = names.nextElement();
            //通过name获取value
            String initParameter = servletContext.getInitParameter(s);
            writer.println(s +"--->"+ initParameter +"<br>");


        }


xml 复制代码
<!--以上两个方法是ServletContext对象的方法,这个方法获取的是什么信息?是以下的配置信息-->
<context-param>
    <param-name>pageSize</param-name>
    <param-value>10</param-value>
</context-param>
<context-param>
    <param-name>startIndex</param-name>
    <param-value>0</param-value>
</context-param>
<!--注意:以上的配置信息属于应用级的配置信息,一般一个项目中共享的配置信息会放到以上的标签当中。-->
<!--如果你的配置信息只是想给某一个servlet作为参考,那么你配置到servlet标签当中即可,使用ServletConfig对象来获取。-->

获取应用的根路径(非常重要),因为在java源代码当中有一些地方可能会需要应用的根路径,这个方法可以动态获取应用的根路径

在java源码当中,不要将应用的根路径写死,因为你永远都不知道这个应用在最终部署的时候,起一个什么名字。

java 复制代码
 获取项目的根路径
 获取应用的根路径(非常重要),因为在java源代码当中有一些地方可能会需要应用的根路径,这个方法可以动态获取应用的根路径
 在java源码当中,不要将应用的根路径写死,因为你永远都不知道这个应用在最终部署的时候,起一个什么名字。
        writer.print("获取项目的根路径:");
        String contextPath = servletContext.getContextPath();
        //  /servlet04
        writer.println(contextPath+"<br>");
java 复制代码
获取文件的绝对路径
 String realPath = servletContext.getRealPath("AServlet.java");
        writer.println(realPath+"<br>");
java 复制代码
  通过ServletContext对象也是可以记录日志的
public void log(String message);
public void log(String message, Throwable t);
  这些日志信息记录到哪里了?
  localhost.2022-03-12.log

  Tomcat服务器的logs目录下都有哪些日志文件?
 catalina.2022-03-12.log 服务器端的java程序运行的控制台信息。
 localhost.2022-03-12.log ServletContext对象的log方法记录的日志信息存储到这个文件中。
 localhost_access_log.2022-03-12.txt 访问日志

应用域(重点)

ServletContext 对象还有另一个名字:应用域(后面还有其他域,例如:请求域、会话域)

如果所有的用户共享一份数据,并且这个数据很少的被修改,并且这个数据量很少,可以将这些数据放到ServletContext这个应用域中

为什么是所有用户共享的数据? 不是共享的没有意义。因为ServletContext这个对象只有一个。只有共享的数据放进去才有意义。

为什么数据量要小? 因为数据量比较大的话,太占用堆内存,并且这个对象的生命周期比较长,服务器关闭的时候,这个对象才会被销毁。大数据量会影响服务器的性能。占用内存较小的数据量可以考虑放进去。

为什么这些共享数据很少的修改,或者说几乎不修改?

所有用户共享的数据,如果涉及到修改操作,必然会存在线程并发所带来的安全问题。所以放在ServletContext对象中的数据一般都是只读的。

数据量小、所有用户共享、又不修改,这样的数据放到ServletContext这个应用域当中,会大大提升效率。因为应用域相当于一个缓存,放到缓存中的数据,下次在用的时候,不需要从数据库中再次获取,大大提升执行效率。

java 复制代码
  存(怎么向ServletContext应用域中存数据)
public void setAttribute(String name, Object value);   map.put(k, v)
  取(怎么从ServletContext应用域中取数据)
public Object getAttribute(String name);   Object v = map.get(k)
  删(怎么删除ServletContext应用域中的数据)
public void removeAttribute(String name);   map.remove(k)
  • 注意:以后我们编写Servlet类的时候,实际上是不会去直接继承GenericServlet类的,因为我们是B/S结构的系统,这种系统是基于HTTP超文本传输协议的,在Servlet规范当中,提供了一个类叫做HttpServlet,它是专门为HTTP协议准备的一个Servlet类。我们编写的Servlet类要继承HttpServlet。(HttpServlet是HTTP协议专用的。)使用HttpServlet处理HTTP协议更便捷。但是你需要知道它的继承结构:
java 复制代码
>    javax.servlet.Servlet(接口)【爷爷】
>     javax.servlet.GenericServlet implements Servlet(抽象类)【儿子】
>     javax.servlet.http.HttpServlet extends GenericServlet(抽象类)【孙子】
>     
>    我们以后编写的Servlet要继承HttpServlet类。

目前为止都接触过哪些缓存机制了?

  • 堆内存当中的字符串常量池。

    • "abc" 先在字符串常量池中查找,如果有,直接拿来用。如果没有则新建,然后再放入字符串常量池。
  • 堆内存当中的整数型常量池。

    • [-128 ~ 127] 一共256个Integer类型的引用,放在整数型常量池中。没有超出这个范围的话,直接从常量池中取。
  • 连接池(Connection Cache)

    • 这里所说的连接池中的连接是java语言连接数据库的连接对象:java.sql.Connection对象。

    • JVM是一个进程。MySQL数据库是一个进程。进程和进程之间建立连接,打开通道是很费劲的。是很耗费资源的。怎么办?可以提前先创建好N个Connection连接对象,将连接对象放到一个集合当中,我们把这个放有Connection对象的集合称为连接池。每一次用户连接的时候不需要再新建连接对象,省去了新建的环节,直接从连接池中获取连接对象,大大提升访问效率。

    • 连接池

      • 最小连接数
      • 最大连接数
      • 连接池可以提高用户的访问效率。当然也可以保证数据库的安全性。
  • 线程池

    • Tomcat服务器本身就是支持多线程的。

    • Tomcat服务器是在用户发送一次请求,就新建一个Thread线程对象吗?

      • 当然不是,实际上是在Tomcat服务器启动的时候,会先创建好N多个线程Thread对象,然后将线程对象放到集合当中,称为线程池。用户发送请求过来之后,需要有一个对应的线程来处理这个请求,这个时候线程对象就会直接从线程池中拿,效率比较高。
      • 所有的WEB服务器,或者应用服务器,都是支持多线程的,都有线程池机制。
  • redis

    • NoSQL数据库。非关系型数据库。缓存数据库。
  • 向ServletContext应用域中存储数据,也等于是将数据存放到缓存cache当中

相关推荐
V+zmm1013428 分钟前
基于微信小程序的乡村政务服务系统springboot+论文源码调试讲解
java·微信小程序·小程序·毕业设计·ssm
Oneforlove_twoforjob1 小时前
【Java基础面试题025】什么是Java的Integer缓存池?
java·开发语言·缓存
xmh-sxh-13141 小时前
常用的缓存技术都有哪些
java
AiFlutter1 小时前
Flutter-底部分享弹窗(showModalBottomSheet)
java·前端·flutter
J不A秃V头A2 小时前
IntelliJ IDEA中设置激活的profile
java·intellij-idea
DARLING Zero two♡2 小时前
【优选算法】Pointer-Slice:双指针的算法切片(下)
java·数据结构·c++·算法·leetcode
小池先生2 小时前
springboot启动不了 因一个spring-boot-starter-web底下的tomcat-embed-core依赖丢失
java·spring boot·后端
CodeClimb2 小时前
【华为OD-E卷-木板 100分(python、java、c++、js、c)】
java·javascript·c++·python·华为od
程序员厉飞雨2 小时前
Android R8 耗时优化
android·java·前端
odng2 小时前
IDEA自己常用的几个快捷方式(自己的习惯)
java·ide·intellij-idea