Servlet ServletConfig接口、ServletContext接口详解

先自我介绍一下,小编浙江大学毕业,去过华为、字节跳动等大厂,目前阿里P7

深知大多数程序员,想要提升技能,往往是自己摸索成长,但自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年最新网络安全全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友。
img
img
img
img
img
img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上网络安全知识点,真正体系化!

由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新

如果你需要这些资料,可以添加V获取:vip204888 (备注网络安全)
img

正文

// 第一种方式:通过ServletConfig对象获取ServletContext对象。
ServletContext application = config.getServletContext();
//org.apache.catalina.core.ApplicationContextFacade@f0fa019
out.print("<br>" + application); 

// 第二种方式:通过this也可以获取ServletContext对象。
ServletContext application2 = this.getServletContext();
//org.apache.catalina.core.ApplicationContextFacade@f0fa019
out.print("<br>" + application2); 

(1)ServletContext是什么?

ServletContext是一个接口,Tomcat服务器对ServletContext接口进行了实现。

(2)ServletContext是谁实现的?

Tomcat服务器(WEB服务器)实现了ServletContext接口。

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

  public class org.apache.catalina.core.ApplicationContextFacade 
implements ServletContext {}

(3)ServletContext对象是谁创建的?在什么时候创建的?

ServletContext对象是WEB服务器创建的。

ServletContext对象WEB服务器启动的时候创建,在服务器关闭的时候销毁!

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

(4)ServletContext怎么理解?

context是什么意思:Servlet对象的环境对象。(Servlet对象的上下文对象
一个ServletContext对象其实对应的就是整个web.xml文件。
理解:50个学生,每个学生都是一个Servlet,这50个学生都在同一个教室当中。那么这个教室就相当于ServletContext对象。并且放在ServletContext对象当中的数据,所有Servlet一定是共享的。
例如:一个教室中的空调是所有学生共享的,一个教室中的语文老师是所有学生共享的。
Tomcat是一个容器,一个容器当中是可以放多个webapp,但是一个webapp只对应一个ServletContext对象。

(5)验证ServletContext对象是共享的!

定义一个Aservlet和一个Bservlet,通过这两个类都继承GenericServlet,然后调用getServletContext()方法获得ServletContext对象,发现是同一个对象。

Aservlet

package com.bjpowernode.javaweb.servlet;

import javax.servlet.*;
import java.io.IOException;
import java.io.PrintWriter;


public class Aservlet extends GenericServlet {
    @Override
    public void service(ServletRequest request, ServletResponse response) throws ServletException, IOException {
        // 设置响应代码类型
        response.setContentType("text/html");
        PrintWriter out = response.getWriter();
        // 获取ServletContext对象
        ServletContext application = this.getServletContext();
        // 输出打印
        // org.apache.catalina.core.ApplicationContextFacade@3d8fcad7
        out.print(application);

    }
}

Bservlet

package com.bjpowernode.javaweb.servlet;

import javax.servlet.*;
import java.io.IOException;
import java.io.PrintWriter;


public class Bservlet extends GenericServlet {
    @Override
    public void service(ServletRequest request, ServletResponse response) throws ServletException, IOException {
        // 设置响应代码类型
        response.setContentType("text/html");
        PrintWriter out = response.getWriter();
        // 获取ServletContext对象
        ServletContext application = this.getServletContext();
        // 输出打印
        // org.apache.catalina.core.ApplicationContextFacade@3d8fcad7
        out.print(application);

    }
}

配置文件web.xml

<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_4_0.xsd"
         version="4.0">
    <servlet>
        <servlet-name>aservlet</servlet-name>
        <servlet-class>com.bjpowernode.javaweb.servlet.Aservlet</servlet-class>
    </servlet>
    <servlet-mapping>
        <servlet-name>aservlet</servlet-name>
        <url-pattern>/a</url-pattern>
    </servlet-mapping>

    <servlet>
        <servlet-name>bservlet</servlet-name>
        <servlet-class>com.bjpowernode.javaweb.servlet.Bservlet</servlet-class>
    </servlet>
    <servlet-mapping>
        <servlet-name>bservlet</servlet-name>
        <url-pattern>/b</url-pattern>
    </servlet-mapping>
</web-app>

执行结果

(6)ServletContext接口中有哪些常用的方法?

①public String getInitParameter(String name)  通过初始化参数的name获取value
②public Enumeration getInitParameterNames() 获取所有的初始化参数的name

ServletContext对象中也有这两个方法,这个两个方法获取的是上下文的初始化参数配置信息;这些配置信息用ServletContext对象来获取。

增加配置信息

注意:

①以下的配置信息属于应用级的配置信息(相当于全局的配置信息),一般一个项目中共享的配置信息会放到以下的**标签当中,使用ServletContext对象来获取
②如果你的配置信息只是想给
某一个servlet**作为参考,那么你配置到 ****标签当中即可,使用ServletConfig对象来获取

 <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>
    

增加通过方法调用获取到配置信息

 // 获取上下文的初始化参数
 Enumeration<String> initParameterNames = application.getInitParameterNames();
 while(initParameterNames.hasMoreElements()){
       // 获取每一个对象
       String name = initParameterNames.nextElement();
       // 通过name获取value
       String value = application.getInitParameter(name);
       out.print(name+"="+value);
       out.print("<br>");
 }
/*
    startIndex=0
    pageSize=10
*/

③public String getContextPath() 获取应用的根路径,相当于拿到了项目名!

// 动态获取context path (获取应用上下文的根)
String contextPath = application.getContextPath();
out.print(contextPath+"<br>"); // "/servlet04"

④public String getRealPath(String path) 获取文件的绝对路径(真实路径)

例如:我们都知道web是根目录,那么就在web根目录下面创建一个common目录;然后在common目录下在创建一个common.html文件。

后面的这个路径,加了一个“/”,这个“/”代表的是web的根;不加“/”,默认也是从根下开始找

// 获取文件的绝对路径,“/”加不加都行,默认都是从web的根下开始找的
String realPath = application.getRealPath("/common/commom.html");
String realPath = application.getRealPath("commom.html");
//C:\Users\86177\IdeaProjects\JavaWeb\out\artifacts\servlet04_war_exploded\common\commom.html
out.print(realPath);

⑤通过ServletContext对象调用下面两个无参方法也是可以记录日志的

public void log(String message) 记录日志的方法
public void log(String message, Throwable t) 记录日志的方法

注意:如果使用文本编译器,这个日志会自动记录到CATALINA_HOME/logs目录下。

如果使用IDEA,IDEA可以创建多个Tomcat服务器;所以日志文件肯定是和IDEA相关的目录下;启动Tomcat找到下面这句话:Using CATALINA_BASE:   “C:\Users\86177.IntelliJIdea2018.3\system\tomcat\Tomcat_9_0_68_JavaWeb”; 这些是参照CATALINA_HOME下的资源生成的一个Tomcat副本:

启动Tomcat服务器会生成两个日志文件:

catalina.2022-11-03.log 服务器端的java程序运行的控制台信息。

catalina.2022-11-03.log 客户端发出请求的访问日志。

客户端发出请求又生成一个日志文件:

localhost.2022-11-03.log ServletContext对象的log方法记录的日志信息存储到这个文件中。

也可以多传一个处理异常的参数!

  int age = 17; // 17岁
  // 当年龄小于18岁的时候,表示非法,记录日志
  if(age < 18) {
     application.log("对不起,您未成年,请绕行!", new RuntimeException("小屁孩,快走开,不适合你!"));
  }

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

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

①为什么是所有用户共享的数据?

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

②为什么数据量要小?

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

③为什么这些共享数据很少的修改,或者说几乎不修改?
所有用户共享的数据,如果涉及到修改操作,必然会存在线程并发所带来的安全问题。所以放在ServletContext对象中的数据一般都是只读的。数据量小、所有用户共享、又不修改,这样的数据放到ServletContext这个应用域当中,会大大提升效率。因为应用域相当于一个缓存,放到缓存中的数据,下次在用的时候,不需要从数据库中再次获取,大大提升执行效率。

(8)怎么向ServletContext应用域中存数据、取数据、删数据?

①存(怎么向ServletContext应用域中存数据)
        public void setAttribute(String name, Object value);
②取(怎么从ServletContext应用域中取数据)
        public Object getAttribute(String name); 
③删(怎么删除ServletContext应用域中的数据)
        public void removeAttribute(String name);

注:很像Map集合(key,value);分别对应着:map.put(k, v)方法、Object v = map.get(k)方法、map.remove(k)方法。

// 准备数据
Date date = new Date();
SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
// 向ServletContext应用域当中存储数据
application.setAttribute("nowDate",sdf.format(date));
// 取出来,输出到浏览器
Object nowDate = application.getAttribute("nowDate");
out.print(nowDate); // 2022-11-03 20:24:23
// 删除数据
application.removeAttribute("nowDate");
out.print(application.getRealPath("nowDate"));

总结

以后编写Servlet类的时候,实际上是不会去直接继承GenericServlet类的,因为B/S结构的系统是基于HTTP超文本传输协议的,在Servlet规范当中,提供了一个类叫做HttpServlet,它是专门为HTTP协议准备的一个Servlet类。编写的Servlet类要直接继承HttpServlet(HttpServlet是HTTP协议专用的)使用HttpServlet处理HTTP协议更便捷。但是需要知道它的继承结构:

jakarta.servlet.Servlet(接口)【爷爷】
jakarta.servlet.GenericServlet implements Servlet(抽象类)【儿子】
jakarta.servlet.http.HttpServlet extends GenericServlet(抽象类)【孙子】

三:补充缓冲机制

到目前为止都接触过的缓存机制!

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

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

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

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

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

    • 连接池

      • 设置最小连接数

本人从事网路安全工作12年,曾在2个大厂工作过,安全服务、售后服务、售前、攻防比赛、安全讲师、销售经理等职位都做过,对这个行业了解比较全面。

最近遍览了各种网络安全类的文章,内容参差不齐,其中不伐有大佬倾力教学,也有各种不良机构浑水摸鱼,在收到几条私信,发现大家对一套完整的系统的网络安全从学习路线到学习资料,甚至是工具有着不小的需求。

最后,我将这部分内容融会贯通成了一套282G的网络安全资料包,所有类目条理清晰,知识点层层递进,需要的小伙伴可以点击下方小卡片领取哦!下面就开始进入正题,如何从一个萌新一步一步进入网络安全行业。

学习路线图

其中最为瞩目也是最为基础的就是网络安全学习路线图,这里我给大家分享一份打磨了3个月,已经更新到4.0版本的网络安全学习路线图。

相比起繁琐的文字,还是生动的视频教程更加适合零基础的同学们学习,这里也是整理了一份与上述学习路线一一对应的网络安全视频教程。

网络安全工具箱

当然,当你入门之后,仅仅是视频教程已经不能满足你的需求了,你肯定需要学习各种工具的使用以及大量的实战项目,这里也分享一份我自己整理的网络安全入门工具以及使用教程和实战。

项目实战

最后就是项目实战,这里带来的是SRC资料&HW资料,毕竟实战是检验真理的唯一标准嘛~

面试题

归根结底,我们的最终目的都是为了就业,所以这份结合了多位朋友的亲身经验打磨的面试题合集你绝对不能错过!

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化的资料的朋友,可以添加V获取:vip204888 (备注网络安全)
img

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

https://img-blog.csdnimg.cn/img_convert/bcd1787ce996787388468bb227d8f959.jpeg)

项目实战

最后就是项目实战,这里带来的是SRC资料&HW资料,毕竟实战是检验真理的唯一标准嘛~

面试题

归根结底,我们的最终目的都是为了就业,所以这份结合了多位朋友的亲身经验打磨的面试题合集你绝对不能错过!

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化的资料的朋友,可以添加V获取:vip204888 (备注网络安全)
[外链图片转存中…(img-GzDtvzdR-1713402172982)]

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

  • 12
    点赞
  • 19
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值