Servlet、HTTP协议、Request

Servlet、HTTP协议、Request🚄


一、Servlet

1. 概念

2. 步骤

3. 执行原理

4. 生命周期

5. Servlet3.0注解配置

6. Servlet体系结构

​ Servlet — 接口

​ |

​ GenericServlet — 抽象类

​ |

​ HttpServlet ---- 抽象类

6.1 GenericServlet(抽象类)

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

将来定义Servlet类时,可以继承GenericServlet抽象类,只需要复写(override)service()方法即可

6.2 HttpServlet(抽象类)【推荐使用】

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-0ScGF4Ec-1629220838855)(E:\POSTGRADUATE\Java\文件\JavaWeb\day14_Servlet&HTTP&Request\截图\HTTP协议.bmp)]

对http协议的封装,简化操作

  1. 定义类继承HttpServlet
  2. 复写doGet()或者doPost()方法

7. Servlet相关配置

  1. urlpattern:Servlet访问路径
    • 一个servlet可以定义多个访问路径
      • @WebServlet({"/demo04","/demo004","/demo0004"})
    • 路径定义规则
      • /xxx
      • /xxx/xxx目录结构
      • *.do

二、HTTP协议

1. 概念

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

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

在这里插入图片描述

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

2. 数据格式

2.1 请求消息数据格式(Request)
  1. 请求行
    • 请求方式 请求url 请求协议/版本
    • GET /login.html HTTP/1.1
    • 请求方式:
      • HTTP协议有8种请求方式,常用的有2中
      • GET:
        • 请求参数在请求行中,在url后【最关键】
        • 请求的url长度是有限制的
        • 不太安全
      • POST:
        • 请求参数在请求体中【最关键】
        • 请求的url长度是没有限制的
        • 相对安全
  2. 请求头(键值对的形式):客户端浏览器告诉服务器的一些信息
    • 请求头名称:请求头值1,请求头值2…
    • 常见的请求头:
      • User-Agent:浏览器告诉服务器,我访问你使用的浏览器版本信息
        • 可以在服务器端获取该头的信息,解决的兼容性问题
      • Referer:http://localhost/login.html
        • 告诉服务器,我(当前请求)从哪里来
        • 作用:
          • 防盗链
          • 统计工作
  3. 请求空行
    • 就是一个空行,用于分割POST请求的请求头和请求体的
  4. 请求体(正文)
    • 封装post请求消息的请求参数的

具体的字符串格式:

​ POST /demo03 HTTP/1.1

​ Host: localhost
​ User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:89.0) Gecko/20100101 Firefox/89.0
​ Accept: text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,/;q=0.8
​ Accept-Language: zh-CN,zh;q=0.8,zh-TW;q=0.7,zh-HK;q=0.5,en-US;q=0.3,en;q=0.2
​ Accept-Encoding: gzip, deflate
​ Content-Type: application/x-www-form-urlencoded
​ Content-Length: 12
​ Origin: http://localhost
​ Connection: keep-alive
​ Referer: http://localhost/login.html
​ Cookie: Pycharm-a508dd0=c0921a79-ae99-4ee8-b754-02641f5de989
​ Upgrade-Insecure-Requests: 1

​ username=zhangsan

2.2 响应消息数据格式(Response)

三、Request

1. request对象和response对象的原理

请添加图片描述

  1. request对象和response对象是由服务器创建的,我们来使用它
  2. request对象是来获取请求消息,response对象是来设置响应消息

2. request对象的集成体系结构

ServletRequest ------- 接口

​ | 继承

HttpServletRequest ------- 接口

​ | 实现

org.apache.catalina.connector.RequestFacade ---- 类(tomcat编写的)

3. request功能

3.1 获取请求消息数据
  1. 获取请求行数据数据
    • GET /day14/demo1?name=zhangsan HTTP/1.1
    • 方法:
      • 获取请求方式:String getMethod()
      • 获取虚拟目录(重点掌握)String getContextPath() —> /day14
      • 获取servlet的路径:String getServletPath() —> /demo
      • 获取get方式的请求参数:String getQueryString() —> name=zhangsan
      • 获取请求的uri(重点掌握)
        • String getRequestURI() —> /day14/demo1
        • StringBuffer getRequestURL() —> http://localhost/day14/demo1
        • URL:统一资源定位符 http://localhost/day14/demo1 中华人民共和国
        • URI:统一资源标识符 /day14/demo1 共和国
      • 获取协议及版本:String getProtocol() —> HTTP/1.1
      • 获取客户机的IP地址:String getRemoteAddr()
  2. 获取请求头数据
    • String getHeader(String name):通过请求头的名称,获取请求头的值(键值对)【重点掌握】
    • Enumeration<String> getHeaderNames():获取所有的请求头名称(当做迭代器使用)
  3. 获取请求体数据
    • 请求体:只有POST请求方式,才有请求体,在请求体中封装了POST请求的请求参数
    • 步骤:
      1. 获取流对象
        1. BufferedReader getReader():获取字符输入流,只能操作字符数据
        2. ServletInputStream getInputStream():获取字节输入流,可以操作任何类型的数据
          • 在文件上传中使用
      2. 再从流对象中拿取数据
3.2 其他功能
  1. 获取请求参数(通用方式)

    1. String getParameter(String name):根据参数名称,获取参数值。如:username=zs&password=123
    2. String[] getParameterValues(String name):根据请求参数的名称获取参数值的数组,如:hobby=xx&hobby=games 一般作用于复选框
    3. Enumberation<String> getParameterNames():获取所有请求的参数名称
    4. Map<String, String[]> getParameterMap():获取所有参数的Map集合
  2. 请求转发:一种在服务器内部的资源跳转方式

在这里插入图片描述

  1. 步骤:

    1. 通过request对象获取请求转发器对象:RequestDispatcher getRequestDispatcher(String path)
    2. 使用这个对象进行转发:void forward(ServletRequest request, ServletResponse response)
    @Override
    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        System.out.println("demo05被访问了");
        /*RequestDispatcher rd = request.getRequestDispatcher("/RequestDemo06");
        rd.forward(request,response);*/
        request.getRequestDispatcher("/RequestDemo06").forward(request, response);
    }
    
  2. 特点 【重要】

    1. 浏览器地址栏路径不发生变化
    2. 只能转发到当前服务器内部资源中
    3. 转发是一次请求(在Chrome浏览器中按F12可以查看)
  3. 共享数据

在这里插入图片描述

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

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

  3. 方法:

    1. setAttribute(String name, Object obj):存储数据
    2. Object getAttribute(String name):通过键获取值
    3. void removeAttribute(String name):通过键移除键值对
  4. 获取ServletContext

    1. ServletContext getServletContext()

4. 案例

用户登录案例需求:
1.编写login.html登录页面
username & password 两个输入框
2.使用Druid数据库连接池技术,操作mysql,day14数据库中user表
3.使用JdbcTemplate技术封装JDBC
4.登录成功跳转到SuccessServlet展示:登录成功!用户名,欢迎您
5.登录失败跳转到FailServlet展示:登录失败,用户名或密码错误

  • 开发步骤:

    1. 创建项目,导入(创建)html页面,配置文件以及jar包…
    2. 创建数据库的环境
    create table users(
        id int primary key auto_increment,
        username varchar(32) unique not null,
        password varchar(32) not null
    );
    
    1. 创建包cn.edu.tjpu.domain,创建类User,对应数据库

      package cn.edu.tjpu.domain;
      
      /**
       * 用户的JavaBean,实体类
       */
      public class User {
          private Integer id;
          private String username;
          private String password;
      
          public User() {
          }
      
          public User(Integer id, String username, String password) {
              this.id = id;
              this.username = username;
              this.password = password;
          }
      
          @Override
          public String toString() {
              return "User{" +
                      "id=" + id +
                      ", username='" + username + '\'' +
                      ", password='" + password + '\'' +
                      '}';
          }
      
          public Integer getId() {
              return id;
          }
      
          public void setId(Integer id) {
              this.id = id;
          }
      
          public String getUsername() {
              return username;
          }
      
          public void setUsername(String username) {
              this.username = username;
          }
      
          public String getPassword() {
              return password;
          }
      
          public void setPassword(String password) {
              this.password = password;
          }
      }
      
      
    2. 创建工具类包cn.edu.tjpu.util,创建JDBCUtils

      package cn.edu.tjpu.util;
      
      import com.alibaba.druid.pool.DruidDataSourceFactory;
      
      import javax.sql.DataSource;
      import java.sql.Connection;
      import java.sql.ResultSet;
      import java.sql.SQLException;
      import java.sql.Statement;
      import java.util.Properties;
      
      public class JDBCUtils {
      
          private static DataSource ds;
      
          static {
              try {
                  Properties prop = new Properties();
                  prop.load(JDBCUtils.class.getClassLoader().getResourceAsStream("druid.properties"));
                  ds = DruidDataSourceFactory.createDataSource(prop);
              } catch (Exception e) {
                  e.printStackTrace();
              }
          }
      
          /**
           * 获取连接Connection对象
           *
           * @return Connection对象
           * @throws SQLException SQL异常
           */
          public static Connection getConnection() throws SQLException {
              return ds.getConnection();
          }
      
          /**
           * 获取DataSource对象
           *
           * @return DataSource
           */
          public static DataSource getDataSource() {
              return ds;
          }
      
          public static void close(Statement stmt, Connection conn) {
              close(null, stmt, conn);
          }
      
          public static void close(ResultSet rs, Statement stmt, Connection conn) {
              if (rs != null) {
                  try {
                      rs.close();
                  } catch (SQLException e) {
                      e.printStackTrace();
                  }
              }
              if (stmt != null) {
                  try {
                      stmt.close();
                  } catch (SQLException e) {
                      e.printStackTrace();
                  }
              }
              if (conn != null) {
                  try {
                      conn.close();
                  } catch (SQLException e) {
                      e.printStackTrace();
                  }
              }
          }
      }
      
    3. 创建包cn.edu.tjpu.dao,创建UserDao,提供login的方法

      package cn.edu.tjpu.dao;
      
      import cn.edu.tjpu.domain.User;
      import cn.edu.tjpu.util.JDBCUtils;
      import org.springframework.dao.DataAccessException;
      import org.springframework.jdbc.core.BeanPropertyRowMapper;
      import org.springframework.jdbc.core.JdbcTemplate;
      
      /**
       * 操作数据库中User表的类
       */
      public class UserDao {
      
          // 声明一个成员变量JDBCTemplate对象来公用
          private JdbcTemplate template = new JdbcTemplate(JDBCUtils.getDataSource());
      
          /**
           * 登录方法
           *
           * @param loginUser 只有用户名和密码
           * @return user包含用户全部数据,没有查询到返回null
           */
          public User login(User loginUser) {
      
              try {
                  // 1. 编写SQL
                  String sql = "select * from users where username=? and password=?";
                  // 2. 调用query()方法
                  User user = template.queryForObject(sql,
                          new BeanPropertyRowMapper<>(User.class),
                          loginUser.getUsername(),
                          loginUser.getPassword());
                  return user;
              } catch (DataAccessException e) {
                  e.printStackTrace();
                  return null;
              }
      
          }
      }
      
    4. 创建cn.edu.tjpu.servlet,创建LoginServlet类

      • login.html中form表单action路径的写法:虚拟目录+Servlet的资源路径
      package cn.edu.tjpu.web.servlet;
      
      import cn.edu.tjpu.dao.UserDao;
      import cn.edu.tjpu.domain.User;
      
      import javax.servlet.*;
      import javax.servlet.http.*;
      import javax.servlet.annotation.*;
      import java.io.IOException;
      
      @WebServlet("/loginServlet")
      public class LoginServlet extends HttpServlet {
          @Override
          protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
              // 1.设置编码
              request.setCharacterEncoding("utf-8");
              // 2.获取请求参数
              String username = request.getParameter("username");
              String password = request.getParameter("password");
              // 3.封装user对象
              User loginUser = new User();
              loginUser.setUsername(username);
              loginUser.setPassword(password);
              // 4.调用UserDao的login方法
              UserDao dao = new UserDao();
              User user = dao.login(loginUser);
              // 5.判断User
              if (user == null) {
                  // 登录失败
                  request.getRequestDispatcher("/failServlet").forward(request, response);
              } else {
                  // 登录成功,存储数据
                  request.setAttribute("user", user);
                  request.getRequestDispatcher("/successServlet").forward(request, response);
              }
          }
      
          @Override
          protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
              this.doGet(request, response);
          }
      }
      
    5. 登录成功跳转至SuccessServlet,登录失败跳转至FailServlet

      package cn.edu.tjpu.web.servlet;
      
      import cn.edu.tjpu.domain.User;
      
      import javax.servlet.*;
      import javax.servlet.http.*;
      import javax.servlet.annotation.*;
      import java.io.IOException;
      
      @WebServlet(name = "SuccessServlet", value = "/successServlet")
      public class SuccessServlet extends HttpServlet {
          @Override
          protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
              // 获取request域中的共享的user对象
              User user = (User) request.getAttribute("user");
      
              if (user != null) {
                  // 登录成功
                  // 设置编码
                  response.setContentType("text/html;charset=utf-8");
                  // 输出
                  response.getWriter().write("登录成功!" +user.getUsername()+",欢迎您");
              }
          }
      
          @Override
          protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
              this.doGet(request, response);
          }
      }
      
      package cn.edu.tjpu.web.servlet;
      
      import javax.servlet.*;
      import javax.servlet.http.*;
      import javax.servlet.annotation.*;
      import java.io.IOException;
      
      @WebServlet(name = "FailServlet", value = "/failServlet")
      public class FailServlet extends HttpServlet {
          @Override
          protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
              // 给页面写一句话
              // 设置编码
              response.setContentType("text/html;charset=utf-8");
              // 输出
              response.getWriter().write("登录失败,用户名或密码错误!");
          }
      
          @Override
          protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
              this.doGet(request, response);
          }
      }
      
    6. login.html中form表单的action路径写法

      • 虚拟路径+Servlet的资源路径
      <!DOCTYPE html>
      <html lang="en">
      <head>
          <meta charset="UTF-8">
          <title>登录界面</title>
      </head>
      <body>
      <form action="/test/loginServlet" method="post">
          <label for="username">用户名:</label>
          <input type="text" name="username" id="username" placeholder="请输入密码">
          <br>
          <label for="password">&nbsp;&nbsp;&nbsp;码:</label>
          <input type="password" name="password" id="password" placeholder="请输入密码">
          <br>
          <button type="submit">登录</button>
      </form>
      </body>
      </html>
      
    7. BeanUtils工具类,简化数据封装

      • 用于封装JavaBean的
      1. JavaBean:标准的Java类
        1. 要求
          1. 类必须被public修饰
          2. 必须提供空参数的构造器
          3. 成员变量必须使用private修饰
          4. 提供public修饰的setter和getter方法
        2. 功能:封装数据
      2. 概念:
        1. 成员变量
        2. 属性:setter和getter方法截取后的产物
          • 例如:getUsername ----> Username ----> username
      3. 方法:
        1. setProperty()
        2. getProperty()
        3. populate(Object obj, Map map):封装JavaBean的方法。将map集合中的键值对信息,封装到对应的JavaBean对象中

【注意事项】

  1. properties文件放到resources路径下
  2. jar包需要放到WEB-INF路径下的lib文件中,且lib不能写为libs
  3. 在JDBCUtils类中,加载properties文件时,一定要与文件名大小写保持一致。因为JDBC测试时,不区分大小写,而tomcat服务器读取时会区分大小写
static {
        try {
            Properties prop = new Properties();
            prop.load(JDBCUtils.class.getClassLoader().getResourceAsStream("druid.properties"));
            ds = DruidDataSourceFactory.createDataSource(prop);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

在这里插入图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值