Java回顾(十五)HTTP 和Request原理

1、HTTP协议

1.1 概念
  • HTTP: Hyper Text Transfer Protocol 超文本传输协议

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

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

1.2 请求消息数据格式
1.2.1 请求行

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

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

  • GET : (1).请求参数在请求行中,在url后 (2). 请求的url长度有限制的 (3).不太安全
  • POST:(1). 请求参数在请求体中 (2). 请求的url长度没有限制的 (3). 相对安全
1.2.2 请求头

请求头:客户端浏览器告诉服务器一些信息
请求头名称: 请求头值
常见的请求头:

* 常见的请求头:
	1. User-Agent:浏览器告诉服务器,我访问你使用的浏览器版本信息
		* 可以在服务器端获取该头的信息,解决浏览器的兼容性问题

	2. Referer:http://localhost/login.html
		* 告诉服务器,我(当前请求)从哪里来?
			* 作用:
				1. 防盗链:
				2. 统计工作:
1.2.3 请求空行

空行,就是用于分割POST请求的请求头,和请求体的。

1.2.4 请求体

请求体(正文):
* 封装POST请求消息的请求参数的

	* 字符串格式:
		POST /login.html	HTTP/1.1
		Host: localhost
		User-Agent: Mozilla/5.0 (Windows NT 6.1; Win64; x64; rv:60.0) Gecko/20100101 Firefox/60.0
		Accept: text/html,application/xhtml+xml,application/xml;q=0.9,*/*;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
		Referer: http://localhost/login.html
		Connection: keep-alive
		Upgrade-Insecure-Requests: 1
		
		username=zhangsan	


* 响应消息数据格式

2、Request

2.1 request对象和response对象的原理

在这里插入图片描述

  1. request和response对象是由服务器创建的。我们来使用它们
  2. request对象是来获取请求消息,response对象是来设置响应消息
2.2 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
			* String getRequestURI():		/day14/demo1
			* StringBuffer getRequestURL()  :http://localhost/day14/demo1

			* URL:统一资源定位符 : http://localhost/day14/demo1	中华人民共和国
			* URI:统一资源标识符 : /day14/demo1					共和国
		
		6. 获取协议及版本:HTTP/1.1
			* String getProtocol()

		7. 获取客户机的IP地址:
			* String getRemoteAddr()
2. 获取请求头数据
	* 方法:
		* (*)String getHeader(String name):通过请求头的名称获取请求头的值
		* Enumeration<String> getHeaderNames():获取所有的请求头名称
	
3. 获取请求体数据:
	* 请求体:只有POST请求方式,才有请求体,在请求体中封装了POST请求的请求参数
	* 步骤:
		1. 获取流对象
			*  BufferedReader getReader():获取字符输入流,只能操作字符数据
			*  ServletInputStream getInputStream():获取字节输入流,可以操作所有类型数据
				* 在文件上传知识点后讲解

		2. 再从流对象中拿数据

这四类方法都很常用

2. 其他功能:
	1. 获取请求参数通用方式:不论get还是post请求方式都可以使用下列方法来获取请求参数
		1. String getParameter(String name):根据参数名称获取参数值    username=zs&password=123
		2. String[] getParameterValues(String name):根据参数名称获取参数值的数组  hobby=xx&hobby=game
		3. Enumeration<String> getParameterNames():获取所有请求的参数名称
		4. Map<String,String[]> getParameterMap():获取所有参数的map集合

		* 中文乱码问题:
			* get方式:tomcat 8 已经将get方式乱码问题解决了
			* post方式:会乱码
				* 解决:在获取参数前,设置request的编码request.setCharacterEncoding("utf-8");
	
			
	2. 请求转发:一种在服务器内部的资源跳转方式
		1. 步骤:
			1. 通过request对象获取请求转发器对象:RequestDispatcher getRequestDispatcher(String path)
			2. 使用RequestDispatcher对象来进行转发:forward(ServletRequest request, ServletResponse response) 

		2. 特点:
			1. 浏览器地址栏路径不发生变化
			2. 只能转发到当前服务器内部资源中。
			3. 转发是一次请求


	3. 共享数据:
		* 域对象:一个有作用范围的对象,可以在范围内共享数据
		* request域:代表一次请求的范围,一般用于请求转发的多个资源中共享数据
		* 方法:
			1. void setAttribute(String name,Object obj):存储数据
			2. Object getAttitude(String name):通过键获取值
			3. void removeAttribute(String name):通过键移除键值对

	4. 获取ServletContext:
		* ServletContext getServletContext()
1、获取请求体数据
@WebServlet("/RequestDemo2")
public class HttpServlet extends javax.servlet.http.HttpServlet {
    @Override
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        BufferedReader br = request.getReader();
        String line = null;
        while ((line = br.readLine()) != null) {
            System.out.println(line);
        }
    }
}

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>注册页面</title>
</head>
<body>
    <form action="/request/RequestDemo2" method="post">
        <input type="text" name="username" placeholder="用户名"><br>
        <input type="text" name="password" placeholder="密码"><br>
        <input type="submit" value="提交">
</form>
</body>
</html>
2、获取所有参数的map集合
@WebServlet("/RequestDemo3")
public class RequestDemo2 extends HttpServlet {
    @Override
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        //获取所有map集合
        Map<String, String[]> parameterMap = request.getParameterMap();
        //遍历
        Set<String> keySet = parameterMap.keySet();
        for (String name:keySet){
            String[] values = parameterMap.get(name);
            System.out.println(name);
            for (String value : values){
                System.out.println(value);
            }
        }
    }
}
3、请求转发
@Override
    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        System.out.println("demo2被访问了");
        request.getRequestDispatcher("/RequestDemo4").forward(request,response);
    }

3、用户登录案例

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

3.1 编写登录页面
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>登录</title>
</head>
<body>
    <form action="/web/loginServlet" method="post">
        <label for="username">用户名</label>
        <input type="text" name="username" id="username"><br>

        <label for="password">密码</label>
        <input type="text" name="password" id="password"><br>

        <input type="submit" value="登录">
    </form>
</body>
</html>
3.2 配置Druid连接池

Druid.properties放在Src的目录下

driverClassName=com.mysql.jdbc.Driver
url=jdbc:mysql:///csdn
username=root
password=123456
initialSize=5
maxActive=10
maxWait=3000

JDBC的工具类

package com.hws.util;

import com.alibaba.druid.pool.DruidDataSourceFactory;

import javax.sql.DataSource;
import javax.xml.crypto.Data;
import java.io.IOException;
import java.io.InputStream;
import java.sql.Connection;
import java.sql.SQLException;
import java.util.Properties;

/**
 * JDBC工具类 使用Durid连接池
 */
public class JDBCUtils {

    private static DataSource ds ;

    static {

        try {
            //1.加载配置文件
            Properties pro = new Properties();
            //使用ClassLoader加载配置文件,获取字节输入流
            InputStream is = JDBCUtils.class.getClassLoader().getResourceAsStream("druid.properties");
            pro.load(is);

            //2.初始化连接池对象
            ds = DruidDataSourceFactory.createDataSource(pro);

        } catch (IOException e) {
            e.printStackTrace();
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    /**
     * 获取连接池对象
     */
    public static DataSource getDataSource(){
        return ds;
    }


    /**
     * 获取连接Connection对象
     */
    public static Connection getConnection() throws SQLException {
        return  ds.getConnection();
    }
}

在UserDao中引入JDBCUtils的getDataSource()

private JdbcTemplate jdbcTemplate = new JdbcTemplate(JDBCUtils.getDataSource());
3.3 创建Bean实体类
package com.hws.domain;

/**
 * ClassName:User
 * Package:com.hws.domain
 * Description:
 *
 * @ date:2020/5/30 15:35
 * @ author:hws
 */
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;
    }

    public User(String username, String password) {
        this.username = username;
        this.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;
    }

    @Override
    public String toString() {
        return "User{" +
                "id=" + id +
                ", username='" + username + '\'' +
                ", password='" + password + '\'' +
                '}';
    }
}

3.4 编写UserDao

先写SQL语句,然后调用query()方法

public class UserDao {

    private JdbcTemplate jdbcTemplate = new JdbcTemplate(JDBCUtils.getDataSource());

    public User login(User loginUser) {
        try {
            //1、编写sql代码
            String sql = "select * from userinfo where username = ? and password = ?";
            //2、调用query
            jdbcTemplate.queryForObject(sql,
                    new BeanPropertyRowMapper<>(User.class),
                    loginUser.getUsername(), loginUser.getPassword());
            return loginUser;
        } catch (DataAccessException e) {
            e.printStackTrace();
            return null;
        }
    }
}

3.5 编写Servlet
@WebServlet("/failServlet")
public class failServlet extends HttpServlet {
    @Override
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        User user = (User) request.getAttribute("user");
        if (user == null){
            response.setContentType("text/html;charset=utf-8");
            response.getWriter().write("您的用户名或密码有误");
        }
    }

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

@WebServlet("/loginServlet")
public class LoginServlet extends HttpServlet {
    @Override
    protected void doPost(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(username,password);

        //4、使用userDao
        UserDao userDao = new UserDao();
        User user = userDao.login(loginUser);

        //5、共享数据
        request.setAttribute("user",user);

        //6、请求转发
        if (user != null){
            request.getRequestDispatcher("/successServlet").forward(request,response);
        }else {
            request.getRequestDispatcher("/failServlet").forward(request,response);
        }
    }

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

@WebServlet("/successServlet")
public class successServlet extends HttpServlet {
    @Override
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        User loginUser = (User) request.getAttribute("user");

        if (loginUser!=null){
            response.setContentType("text/html;charset=utf-8");
            response.getWriter().write("登录成功"+loginUser.getUsername()+"欢迎您");
        }
    }

    @Override
    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        this.doPost(request, response);
    }
}
3.6 拓展
3.6.1 使用BeanUtils包(apache下的)

使用BeanUtils包可以将Map封装到Bean中,页面的信息元素放入Map中,map再通过BeanUtils变成Bean对象

@Override
    protected void doPost(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(username,password);

        //2、获取页面信息的Map集合
        Map<String, String[]> parameterMap = request.getParameterMap();

        User loginUser = new User();

        //3、使用BeanUtils
        try {
            BeanUtils.populate(loginUser,parameterMap);
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        } catch (InvocationTargetException e) {
            e.printStackTrace();
        }

        //4、使用userDao
        UserDao userDao = new UserDao();
        User user = userDao.login(loginUser);

        //5、共享数据
        request.setAttribute("user",user);

        //6、请求转发
        if (user != null){
            request.getRequestDispatcher("/successServlet").forward(request,response);
        }else {
            request.getRequestDispatcher("/failServlet").forward(request,response);
        }
    }
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

隐形的稻草人HWS

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值