loading...
Servlet
Published in:2022-01-31 | category: JavaWeb基础知识总结
Words: 2.6k | Reading time: 10min | reading:

Servlet

概念

运行服务器端的小程序

  • servlet就是一个接口,定义了Java类被浏览器访问到(tomcat识别)的规则
  • 将来我们自定义一个类,实现Servlet接口,重写方法。

快速入门

  1. 创建JavaEE项目

  2. 定义一个类,实现Servlet接口或继承HttpServlet类

  3. 实现接口中的抽象方法

  4. 配置Servlet

Servlet两种创建方式

实现接口Servlet

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
/**
* Servlet创建的第一种方式:实现接口Servlet
**/
public class ServletDemo1 implements Servlet{
public void destory(){
}

public ServletConfig getServletConfig(){
return null;
}

public String getServletInfo(){
return null;
}

public void init(ServletConfig arg0) throws ServletException{
}

public void service(ServletRequest request,ServletResonse response) throws ServletException,IOException{
System.out.println("OK");
response.getWriter().println("welcome use servlet");
}
}
  • 该方式比较麻烦,需要实现接口种所有方法。

继承HttpServlet

1
2
3
4
5
6
7
8
9
10
11
12
/**
* Servlet implementation class HelloServlet
* Servlet的第二种创建方式,继承HttpServlet.也是开发中推荐的
**/
public class ServletDemo1 extends HttpServlet {
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
response.getWriter().print("welcome use servlet");
}
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
doGet(request, response);
}
}

配置Servlet

1
2
3
4
5
6
7
8
9
<!--配置Servlet-->
<servlet>
<servlet-name>ServletDemo1</servlet-name>
<servlet-class>com.yr.web.servlet.ServletDemo1</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>ServletDemo1</servlet-name>
<url-pattern>/demo1</url-pattern>
</servlet-mapping>
  • urlpartten:Servlet的访问路径
    • 一个Servlet可以定义多个访问路径:@WebServlet({“d4”,”dd4”,”ddd4”})
    • 路径定义规则
      • /xxx
      • /xxx/xxx:多层路径,目录结构
      • *.do:任意XXXX.do皆可访问该网页

执行原理

  1. 当服务器接收到客户端浏览器的请求后,会解析请求URL路径,获取访问的Servlet的资源路径
  2. 查找web.xml文件,是否有对应的< url-pattern >标签体内容
  3. 如果有,则在找到对应的< sevrlet-class >全类名
  4. tomcat会将字节码文件加载进内存,并创建其对象
  5. 调用其方法

生命周期

  1. 被创建:执行init方法,只执行一次

    • 默认情况下,第一次被访问时,Servlet被创建

    • 可以配置执行Servlet的创建时机

      • 在< servlet >标签下配置

        • 第一次被访问时,创建

          <load-on-startup>的值为负数

        • 在服务器启动时,创建

          <load-on-startup>的值为0或正整数

    • init方法只执行一次,说明Servlet在内存中只存在一个对象,Servlet是单例的

      • 多个用户同时访问时,可能存在线程安全问题
      • 解决方案:尽量不要在Servlet中定义成员变量。即使定义了成员变量,也不要对其修改值。
  2. 提供服务:执行Service方法,执行多次

    • 每次访问Servlet时,Servlet方法都会被调用一次
  3. 被销毁:执行destory方法,执行多次

    • Servlet被销毁时执行。服务器关闭时,Servlet被销毁
    • 只有服务器正常关闭时,才会被执行destory方法
    • destory方法在Servlet被销毁之前执行,一般用于释放资源

Servlet 3.0

好处

  • 支持注解配置。可以不需要web.xml了

步骤

  1. 创建JavaEE项目,选择Servlet的版本3.0以上,可以不创建web.xml

  2. 定义一个类,实现Servlet接口或继承HTTPServlet类

  3. 复写方法

  4. 在类上使用注解进行配置

    @WebServlet(“资源路径”)

IDEA和TomCat的相关配置

  1. IDEA会为每一个tomcat部署的项目单独建立一份配置文件

    查看控制台的log:Using CATALINA_BASE: “C:\Users\Administrator\AppData\Local\JetBrains\IntelliJIdea2020.2\tomcat\Unnamed_Servlet”

  2. 工作空间项目 和 tomcat部署的web项目

    • tomcat真正访问的是”tomcat部署的web项目”,”tomcat部署的web项目”对应着”工作空间项目”的Web目录下的所有资源

    • WEB-INF下的文件不能直接被浏览器访问。

HttpServlet的基本概念

GenericServlet和HttpServlet

GenericServlet

将Servlet接口中其他的方法做了默认空实现,只将service()方法作为抽象方法

将来定义Servlet类时,可以继承GenericServlet,实现service()方法即可

HttpServlet

对http协议的一种封装,简化操作
1.定义类继承HttpServlet
2.复写doGet/doPost方法

HTTP

概念

HTTP(Hyper Text Transfer Protocol)超文本传输协议:

传输协议定义了客户端和服务端通信时,发送数据的格式

特点

  • 基于TCP/IP的高级协议
  • 默认端口号:80
  • 基于请求/响应模型的:一次请求对应一次相应
  • 无状态的:每次请求之间相互独立,不能交互数据

请求消息数据格式

请求行

请求方式 请求url 请求协议/版本
GET/login.html HTTP/1.1

HTTP协议中有7种请求方式,常用的有2种:

  • GET
    • 请求参数在请求行中,在url后。
    • 请求的url长度是有限制的
    • 不太安全
  • POST
    • 请求参数在请求体中。
    • 请求的url长度没有限制的
    • 相对安全

请求头

作用:客户端浏览器告诉服务器一些信息

请求头名称:请求头值

常见的请求头:

  • User-Agent: 浏览器告诉服务器,我访问你使用的浏览器版本信息。可以在服务器端获取该头的信息,解决浏览器的兼容问题。
  • Referer:告诉服务器,我(当前请求)从哪里来?
    作用:
    • 防盗链
    • 统计工作

请求空行

请求空行其本质就是一行空行,它的作用是分隔POST的请求头和请求体。

请求体(正文)

封装POST请求消息的请求参数。

响应消息

服务器端发送给客户端的数据

响应行

  • 组成:协议/版本 响应状态码 状态码描述
  • 响应状态码:服务器告诉客户端浏览器本次请求和响应的一个状态
    • 状态码都是3位数字
      • 1xx:服务器接收客户端消息,但没有接收完成,等待一段时间后,发送1xx多状态码
      • 2xx:成功。代表:200
      • 3xx:重定向。代表:302(重定向),304访问缓存
      • 4xx:客户端错误。404(路径没有对应的资源) 405(请求方式没有对应的doxxx方法)
      • 5xx:服务器错误。500(服务器端发生异常)

响应头

  • 格式: 头名称: 值
  • 常见的响应头:
    • Content-Type: 服务器告诉客户端本次响应体数据格式以及编码格式
    • Content-disposition:服务器告诉客户端以什么格式打开响应体数据
      • 值:
        • in-line:默认值,在当前页面内打开
        • attachment;filename=xxx:以附件形式打开响应体。文件下载

响应空行

响应空行其本质就是一行空行,它的作用是分隔POST的响应头和响应体。

响应体

封装POST响应消息的响应参数。

Request对象和Response对象

request和response对象是由服务器创建的。

request对象是用来获取请求信息,response对象是用来设置响应信息。

Request

获取请求消息。

Requset继承体系结构
  • ServletRequest –接口
  • HttpServletRequest –接口
  • org.apache.catalina.connector.RequestFacade 类(tomcat)
功能
  • 获取请求行数据

    Get /demo1?name=zhangsan HTTP/1.1

    方法:

    1. 获取请求方式:GET

      String getMethod()

    2. 获取虚拟目录:(重要)
      String getContextPath()

    3. 获取Servlet路径: /demo1
      String getServletPath()

    4. 获取get方式请求参数:name=zhansan
      String getQueryString()

    5. 获取请求方式的URI: (重要)
      String getRequestURI(): /demo1
      StringBuffer getRequestURL(): http://localhost/MyServlet/demo1

    6. 获取协议及版本:Http/1.1
      String getProtocol()

    7. 获取客户机的IP地址
      String getRemoteAddr()

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {    
//1.获取请求方式 /GET
String method = req.getMethod();
System.out.println(method);
//2.获取虚拟目录
String contextPath = req.getContextPath(); System.out.println(contextPath);
//3.获取Servlet路径 / /RequestDemo1
String servletPath = req.getServletPath();
System.out.println(servletPath);
//4.获取get方式请求参数 /null
String queryString = req.getQueryString(); System.out.println(queryString);
//5.获取请求方式的URI和URL /URI:/RequestDemo1 /URL:http://localhost:8080/RequestDemo1
String requestURI = req.getRequestURI();
StringBuffer requestURL = req.getRequestURL(); System.out.println(requestURI);
System.out.println(requestURL);
//6.获取协议及版本 /HTTP/1.1
String protocol = req.getProtocol(); System.out.println(protocol);
//7.获取客户机的IP地址 /0:0:0:0:0:0:0:1
String remoteAddr = req.getRemoteAddr();
System.out.println(remoteAddr);}

URI和URL的区别?

URL:统一资源定位符:http://localhost:8080/RequestDemo1 相当于中华人民共和国

URI:统一资源标识符:/RequestDemo1 相当于共和国

  • 获取请求头数据

    方法:

    ​ (*)String getHeader(String name):通过请求头的名称获取请求头的值

    ​ Enumeration< String > getHeaderNames():获取所有的请求头名称

  • 获取请求体数据
    请求体:只有POST请求方式,才有请求体,在请求体中封装了POST请求的请求参数

    步骤:

    1. 获取流对象
      BufferedReader getReader():获取字符输入流,只能操作字符数据
      ServletInputStream getInputStream():获取字节输入流,可以操作所有类型数据
    2. 再从流对象中获取数据
  • 其他功能

    • 获取请求参数通用方式(不论get还是post请求方式都可以使用下列方法来获取请求参数)
    1
    2
    3
    4
    String getParameter(String name): //根据参数名称获取参数值
    String[] getParameterValues(String name)//根据参数获取参数值的数组
    Enumeration< String > getParameterNames(): //获取所有请求参数的名称
    Map<String ,String[]> getParameterMap(): //获取所有参数的map集合
    • 请求转发:一种在服务器内部的资源跳转方式

      1. 步骤:
        1. 通过request对象获取请求转发器对象:RequestDispatcher getRequestDispatcher(String path)
        2. 使用RequestDispatcher对象来进行转发:forward(ServletRequest request , ServletResponse response)
      2. 特点:
        1. 浏览器地址栏路径不发生变化
        2. 只能转发到当前服务器内部资源
        3. 转发是一次请求
    • 共享数据

      • 域对象:一个有作用范围的对象,可以在范围内共享数据

      • request域:代表一次请求的范围,一般用于请求转发的多个资源中共享数据

      • 方法

        1
        2
        3
        setAttribute(String name ,Object obj):  //存储数据
        Object getAttitude(String name): //通过键获取值
        removeAttribute(String name): //通过键移除一个键值对
    • 获取ServletContext

    ServletContext getServletContext()

    返回对调用者在其中执行操作的ServletContext的引用,返回值为ServletContext对象

Response

设置响应消息

功能
  • 设置响应行

    格式:HTTP/1.1 200 ok
    设置状态码: ```setStatus(int sc)```
    
  • 设置响应头:setHeader(String name ,String value)

  • 设置响应体:

    使用步骤:

    1. 获取输出流

      1
      2
      字符输出流 : PrintWriter getWrite()
      字节输出流: ServletOutputStream getOutputStream()
    2. 使用输出流,将数据输出到客户端浏览器

Prev:
会话技术
Next:
JSP
catalog
catalog