🏆 HTTP

image.png

  1. 概念:Hyper Text Transfer Protocol 超文本传输协议
  2. 传输协议:定义了,客户端和服务器端通信时,发送数据的格式
  3. 特点:
    1. 基于TCP/IP的高级协议
    2. 默认端口号:80
    3. 基于请求/响应模型的:一次请求对应一次响应,如上图。
    4. 无状态的:每次请求之间相互独立,不能交互数据
  4. 历史版本:

1.0:每一次请求响应都会建立新的连接,然后断开连接
1.1:复用连接

  1. public void service(ServletRequest servletRequest, ServletResponse servletResponse) throws ServletException, IOException {
  2. System.out.println("hello,servlet");
  3. }

请求消息数据格式

  1. POST /login.html HTTP/1.1 //请求行
  2. Host: localhost //请求头
  3. User-Agent: Mozilla/5.0 (Windows NT 6.1; Win64; x64; rv:60.0) Gecko/20100101 Firefox/60.0
  4. Accept: text/html,application/xhtml+xml,application/xml;q=0.9,/;q=0.8
  5. 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
  6. Accept-Encoding: gzip, deflate
  7. Referer: http://localhost/login.html
  8. Connection: keep-alive
  9. Upgrade-Insecure-Requests: 1
  10. username=zhangsan //请求体

get请求只有请求行和请求头

请求消息格式详解

  1. 请求行

    1. 请求方式 请求url 请求协议/版本
      1. GET /login.html HTTP/1.1
    2. 请求方式: HTTP协议有7中请求方式,常用的有2种
      1. GET:
        1. 请求参数在请求行中,在url后。
        2. 请求的url长度有限制的
        3. 不太安全
      2. POST:
        1. 请求参数在请求体中
        2. 请求的url长度没有限制的
        3. 相对安全
  2. 请求头:客户端浏览器告诉服务器一些信息

请求头名称: 请求头值,如:

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

image.png

  1. 请求空行

    1. 空行,就是用于分割POST请求的请求头,和请求体的 。
  2. 请求体(正文)

    1. 封装POST请求消息的请求参数的

🏹 request对象和response对象的原理

image.png

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

🎱 request对象

request对象继承体系结构

ServletRequest — 接口
| 继承
HttpServletRequest — 接口
| 实现
org.apache.catalina.connector.RequestFacade 类(tomcat)

request对象的功能

获取请求消息数据

1、获取请求行数据

GET /day14/demo1?name=zhangsan HTTP/1.1

方法:

  1. 获取请求方式 :GET
    • String getMethod()
  2. 获取虚拟目录:/day14
    • String getContextPath()
  3. 获取Servlet路径: /demo1
    • String getServletPath()
  4. 获取get方式请求参数:name=zhangsan
    • String getQueryString()
  5. 获取请求URI:/day14/demo1
  6. 获取协议及版本:HTTP/1.1
    • String getProtocol()
  7. 获取客户机的IP地址:
    • String getRemoteAddr()
      1. //1. 获取请求方式 :GET
      2. String method = request.getMethod();
      3. System.out.println(method);
      4. //2.(*)获取虚拟目录:/day14
      5. String contextPath = request.getContextPath();
      6. System.out.println(contextPath);
      7. //3. 获取Servlet路径: /demo1
      8. String servletPath = request.getServletPath();
      9. System.out.println(servletPath);
      10. //4. 获取get方式请求参数:name=zhangsan
      11. String queryString = request.getQueryString();
      12. System.out.println(queryString);
      13. //5.(*)获取请求URI:/day14/demo1
      14. String requestURI = request.getRequestURI();
      15. StringBuffer requestURL = request.getRequestURL();
      16. System.out.println(requestURI);
      17. System.out.println(requestURL);
      18. //6. 获取协议及版本:HTTP/1.1
      19. String protocol = request.getProtocol();
      20. System.out.println(protocol);
      21. //7. 获取客户机的IP地址:
      22. String remoteAddr = request.getRemoteAddr();
      23. System.out.println(remoteAddr);

2、获取请求头数据

方法:

  1. String getHeader(String name):通过请求头的名称获取请求头的值
  2. Enumeration getHeaderNames():获取所有的请求头名称

    1. protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
    2. //1.获取所有请求头的名称
    3. Enumeration<String> headerNames = request.getHeaderNames();
    4. //2.遍历
    5. while (headerNames.hasMoreElements()) {
    6. String name = headerNames.nextElement();
    7. String value = request.getHeader(name);
    8. //根据名称获取请求头的值
    9. System.out.println(name + "___" + value);
    10. }
    11. }
    1. protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
    2. //获取请求头数据
    3. String agent = request.getHeader("user-agent");
    4. //判断agent的浏览器版本
    5. if (agent.contains("Chrome")){
    6. System.out.println("谷歌浏览器");
    7. }else if (agent.contains("Firefox")){
    8. System.out.println("火狐浏览器");
    9. }
    10. }
    1. protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
    2. String referer = request.getHeader("referer");
    3. System.out.println(referer);
    4. //防盗链
    5. if (referer != null) {
    6. if (referer.contains("/day13")){
    7. System.out.println("正常访问");
    8. //response.setContenType("text/html;charset=utf-8");
    9. //response.getWriter().write("播放电影...");
    10. } else {
    11. System.out.println("想看电影吗?");
    12. }
    13. }
    14. }

3、获取请求体数据

请求体:只有POST请求方式,才有请求体,在请求体中封装了POST请求的请求参数
步骤:

  1. 获取流对象
    • BufferedReader getReader():获取字符输入流,只能操作字符数据
    • ServletInputStream getInputStream():获取字节输入流,可以操作所有类型数据
      • 在文件上传知识点后讲解
  2. 再从流对象中拿数据

    1. protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
    2. //获取请求消息体--请求参数
    3. //1.获取字符流
    4. BufferedReader br = request.getReader();
    5. //2.读取数据
    6. String line = null;
    7. while ((line = br.readLine()) != null) {
    8. System.out.println(line);
    9. }
    10. //注意,写在doPost里面
    11. }

获取请求参数通用方式

不论get还是post请求方式都可以使用下列方法来获取请求参数:

  1. String getParameter(String name): 根据参数名称获取参数值 username=zs&password=123 ```java protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { //post 获取请求参数

    //根据参数名称获取参数值 String username = request.getParameter(“username”); System.out.println(“post”); System.out.println(username); }

protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { //get 获取请求参数 / //根据参数名称获取参数值 String username = request.getParameter(“username”); System.out.println(“get”); System.out.println(username);/

  1. this.doPost(request,response);//直接缩写,省略一堆代码

}

  1. 2. `String[] getParameterValues(String name)`:根据参数名称获取参数值的数组 hobby=xx&hobby=game多用于复选框
  2. ```java
  3. //根据参数名称获取参数值的数组
  4. String[] hobbies = request.getParameterValues("hobby");
  5. for (String hobby : hobbies) {
  6. System.out.println(hobby);
  7. }
  1. Enumeration getParameterNames():获取所有请求的参数名称

    1. //获取所有请求的参数名称
    2. Enumeration<String> parameterNames = request.getParameterNames();
    3. while(parameterNames.hasMoreElements()){
    4. String name = parameterNames.nextElement();//获取参数名
    5. System.out.println(name);
    6. String value = request.getParameter(name);//根据参数名获取参数值
    7. System.out.println(value);
    8. System.out.println("----------------");
    9. }
  2. Map<String,String[]> getParameterMap():获取所有参数的map集合 ```java // 获取所有参数的map集合 Map parameterMap = request.getParameterMap(); //遍历 Set keyset = parameterMap.keySet();

for (String name : keyset) {

  1. //获取键获取值
  2. String[] values = parameterMap.get(name);
  3. System.out.println(name);
  4. for (String value : values) {
  5. System.out.println(value);
  6. }
  7. System.out.println("-----------------");

}

  1. <a name="PbUlr"></a>
  2. #### 中文乱码问题
  3. - get方式:tomcat 8 已经将get方式乱码问题解决了
  4. - post方式:(会乱码 tomcat10也解决了)
  5. - 解决:在获取参数前,设置request的编码`request.setCharacterEncoding("utf-8");`
  6. <a name="TYMx0"></a>
  7. ### 请求转发
  8. > 一种在服务器内部的资源跳转方式
  9. ![image.png](https://cdn.nlark.com/yuque/0/2022/png/21490994/1651310816553-400c1768-ea63-49c4-b1c6-a4f457c4503b.png#clientId=u8b4b9bed-3143-4&crop=0&crop=0&crop=1&crop=1&from=paste&height=546&id=u2bd873be&margin=%5Bobject%20Object%5D&name=image.png&originHeight=1092&originWidth=2378&originalType=binary&ratio=1&rotation=0&showTitle=false&size=241964&status=done&style=none&taskId=ua90f5c7c-b32f-43dd-8c56-f39ac8ca290&title=&width=1189)
  10. - 步骤:
  11. - 1、通过request对象获取请求转发器对象
  12. - `RequestDispatcher getRequestDispatcher(String path)`
  13. - 2、使用RequestDispatcher对象来进行转发
  14. - `forward(ServletRequest request, ServletResponse response) `
  15. - 特点:
  16. - 1、浏览器地址栏路径不发生变化
  17. - 2、只能转发到当前服务器内部资源中,不能转发到网络资源
  18. - 3、转发只是一次请求 ,使用的是同一次请求
  19. ```java
  20. protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
  21. System.out.println("demo8888被访问了。。。");
  22. //转发到demo9资源
  23. /*
  24. RequestDispatcher requestDispatcher = request.getRequestDispatcher("/requestDemo9");
  25. requestDispatcher.forward(request,response);
  26. */
  27. request.getRequestDispatcher("./requestDemo09").forward(request,response);
  28. }

共享数据

image.png

  • 域对象:一个有作用范围的对象,可以在范围内共享数据
  • request域:代表一次请求的范围,一般用于请求转发的多个资源中共享数据
  • 方法:

    • void setAttribute(String name,Object obj):存储数据
    • Object getAttitude(String name):通过键获取值
    • void removeAttribute(String name):通过键移除键值对
    • Enumeration getAttributeNames():返回Enumeration 对象,包含了ServletContext中所有域属性名 ```java protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { System.out.println(“demo08被访问了…”);

      //存储数据到request域中 request.setAttribute(“msg”,”hello”);

      request.getRequestDispatcher(“/requestDemo09”).forward(request,response); }

  1. ```java
  2. protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
  3. //获取数据
  4. Object msg = request.getAttribute("msg");
  5. System.out.println(msg);
  6. System.out.println("demo09被访问了...");
  7. }

获取ServletContext

ServletContext对象封装了当前Web应用的所有信息,而且实现了多个Servlet之间数据共享。

ServletContext getServletContext()返回一个ServletContext对象

  1. protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
  2. ServletContext servletContext = request.getServletContext();
  3. System.out.println(servletContext); //org.apache.catalina.core.ApplicationContextFacade@b91e38f
  4. }

🎿 案例:用户登录

用户登录案例需求

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

分析

image.png

开发步骤

  1. 创建项目,导入html页面,配置文件,jar包
  2. 创建数据库环境

    1. CREATE DATABASE day14;
    2. USE day14;
    3. CREATE TABLE USER(
    4. id INT PRIMARY KEY AUTO_INCREMENT,
    5. username VARCHAR(32) UNIQUE NOT NULL,
    6. PASSWORD VARCHAR(32) NOT NULL,
    7. );
  3. 创建包cn.itcast.domain,创建类User

    1. package cn.itcast.domain;
    2. /**
    3. ■ 用户的实体类
    4. */
    5. public class User {
    6. private int id;
    7. private String username;
    8. private String password;
    9. public int getId() {
    10. return id;
    11. }
    12. public void setId(int id) {
    13. this.id = id;
    14. }
    15. public String getUsername() {
    16. return username;
    17. }
    18. public void setUsername(String username) {
    19. this.username = username;
    20. }
    21. public String getPassword() {
    22. return password;
    23. }
    24. public void setPassword(String password) {
    25. this.password = password;
    26. }
    27. @Override
    28. public String toString() {
    29. return "User{" +
    30. "id=" + id +
    31. ", username='" + username + '\'' +
    32. ", password='" + password + '\'' +
    33. '}';
    34. }
    35. }
  1. 创建包cn.itcast.util,编写工具类JDBCUtils

    1. package cn.itcast.util;
    2. import com.alibaba.druid.pool.DruidDataSourceFactory;
    3. import javax.sql.DataSource;
    4. import javax.xml.crypto.Data;
    5. import java.io.IOException;
    6. import java.io.InputStream;
    7. import java.sql.Connection;
    8. import java.sql.SQLException;
    9. import java.util.Properties;
    10. /**
    11. ○ JDBC工具类 使用Durid连接池
    12. */
    13. public class JDBCUtils {
    14. private static DataSource ds ;
    15. static {
    16. try {
    17. //1.加载配置文件
    18. Properties pro = new Properties();
    19. //使用ClassLoader加载配置文件,获取字节输入流
    20. InputStream is = JDBCUtils.class.getClassLoader().getResourceAsStream("druid.properties");
    21. pro.load(is);
    22. //2.初始化连接池对象
    23. ds = DruidDataSourceFactory.createDataSource(pro);
    24. } catch (IOException e) {
    25. e.printStackTrace();
    26. } catch (Exception e) {
    27. e.printStackTrace();
    28. }
    29. }
    30. /**
    31. ■ 获取连接池对象
    32. */
    33. public static DataSource getDataSource(){
    34. return ds;
    35. }
    36. /**
    37. * 获取连接Connection对象
    38. */
    39. public static Connection getConnection() throws SQLException {
    40. return ds.getConnection();
    41. }
    42. }
  2. 创建包cn.itcast.dao,创建类UserDao,提供login方法 ```java package cn.itcast.dao;

import cn.itcast.domain.User; import cn.itcast.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. //1.编写sql
      2. String sql = "select * from user where username = ? and password = ?";
      3. //2.调用query方法
      4. User user = template.queryForObject(sql,new BeanPropertyRowMapper<User>(User.class),loginUser.getUsername(), loginUser.getPassword());
      5. return user;
      6. } catch (DataAccessException e) {
      7. e.printStackTrace();//记录日志
      8. return null;
      9. }
      10. }
      } ```
  1. 编写cn.itcast.web.servlet.LoginServlet类 ```java package cn.itcast.web.servlet;

import cn.itcast.dao.UserDao; import cn.itcast.domain.User;

import javax.servlet.ServletException; import javax.servlet.annotation.WebServlet; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.io.IOException;

@WebServlet(“/loginServlet”) public class LoginServlet extends HttpServlet { @Override protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { //1.设置编码 req.setCharacterEncoding(“utf-8”); //2.获取请求参数 String username = req.getParameter(“username”); String password = req.getParameter(“password”); //3.封装user对象 User loginUser = new User(); loginUser.setUsername(username); loginUser.setPassword(password);

  1. //4.调用UserDao的login方法
  2. UserDao dao = new UserDao();
  3. User user = dao.login(loginUser);
  4. //5.判断user
  5. if(user == null){
  6. //登录失败
  7. req.getRequestDispatcher("/failServlet").forward(req,resp);
  8. }else{
  9. //登录成功
  10. //存储数据
  11. req.setAttribute("user",user);
  12. //转发
  13. req.getRequestDispatcher("/successServlet").forward(req,resp);
  14. }

}

@Override protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { this.doGet(req,resp); } }

  1. 7. 编写FailServletSuccessServlet
  2. ```java
  3. @WebServlet("/successServlet")
  4. public class SuccessServlet extends HttpServlet {
  5. protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
  6. //获取request域中共享的user对象
  7. User user = (User) request.getAttribute("user");
  8. if(user != null){
  9. //给页面写一句话
  10. //设置编码
  11. response.setContentType("text/html;charset=utf-8");
  12. //输出
  13. response.getWriter().write("登录成功!"+user.getUsername()+",欢迎您");
  14. }
  15. }
  16. @WebServlet("/failServlet")
  17. public class FailServlet extends HttpServlet {
  18. protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
  19. //给页面写一句话
  20. //设置编码
  21. response.setContentType("text/html;charset=utf-8");
  22. //输出
  23. response.getWriter().write("登录失败,用户名或密码错误");
  24. }
  25. protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
  26. this.doPost(request,response);
  27. }
  28. }
  1. login.html中form表单的action路径的写法
    * 虚拟目录+Servlet的资源路径
    1. BeanUtils工具类,简化数据封装
      • 用于封装JavaBean的
      1. JavaBean:标准的Java类
        1. 要求:
          1. 类必须被public修饰
          2. 必须提供空参的构造器
          3. 成员变量必须使用private修饰
          4. 提供公共setter和getter方法
        2. 功能:封装数据
      2. 概念:
        成员变量:
        属性:setter和getter方法截取后的产物
        例如:getUsername() —> Username—> username
      3. 方法:
        1. setProperty()
        2. getProperty()
        3. populate(Object obj , Map map):将map集合的键值对信息,封装到对应的JavaBean对象中