http协议及request

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

        * 历史版本:
            * 1.0:每一次请求响应都会建立新的连接
            * 1.1:复用连接

    * 请求消息数据格式
        1. 请求行
            请求方式 请求url 请求协议/版本
            GET /login.html    HTTP/1.1

            * 请求方式:
                * HTTP协议有7中请求方式,常用的有2种
                    * GET:
                        1. 请求参数在请求行中,在url后。
                        2. 请求的url长度有限制的
                        3. 不太安全
                    * POST:
                        1. 请求参数在请求体中
                        2. 请求的url长度没有限制的
                        3. 相对安全
        2. 请求头:客户端浏览器告诉服务器一些信息
            请求头名称: 请求头值
            * 常见的请求头:
                1. User-Agent:浏览器告诉服务器,我访问你使用的浏览器版本信息
                    * 可以在服务器端获取该头的信息,解决浏览器的兼容性问题

                2. Referer:http://localhost/login.html
                    * 告诉服务器,我(当前请求)从哪里来?
                        * 作用:
                            1. 防盗链:
                            2. 统计工作:
        3. 请求空行
            空行,就是用于分割POST请求的请求头,和请求体的。
        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    

 

 Response对象
    * 功能:设置响应消息
        1. 设置响应行
            1. 格式:HTTP/1.1 200 ok
            2. 设置状态码:setStatus(int sc) 
        2. 设置响应头:setHeader(String name, String value) 
            
        3. 设置响应体:
            * 使用步骤:
                1. 获取输出流
                    * 字符输出流:PrintWriter getWriter()

                    * 字节输出流:ServletOutputStream getOutputStream()

                2. 使用输出流,将数据输出到客户端浏览器


    * 案例:
        1. 完成重定向
            * 重定向:资源跳转的方式
            * 代码实现:
                //1. 设置状态码为302
                response.setStatus(302);
                //2.设置响应头location
                response.setHeader("location","/day15/responseDemo2");


                //简单的重定向方法
                response.sendRedirect("/day15/responseDemo2");

            * 重定向的特点:redirect
                1. 地址栏发生变化
                2. 重定向可以访问其他站点(服务器)的资源
                3. 重定向是两次请求。不能使用request对象来共享数据
            * 转发的特点:forward
                1. 转发地址栏路径不变
                2. 转发只能访问当前服务器下的资源
                3. 转发是一次请求,可以使用request对象来共享数据
            
            * forward 和  redirect 区别
                
            * 路径写法:
                1. 路径分类
                    1. 相对路径:通过相对路径不可以确定唯一资源
                        * 如:./index.html
                        * 不以/开头,以.开头路径

                        * 规则:找到当前资源和目标资源之间的相对位置关系
                            * ./:当前目录
                            * ../:后退一级目录
                    2. 绝对路径:通过绝对路径可以确定唯一资源
                        * 如:http://localhost/day15/responseDemo2        /day15/responseDemo2
                        * 以/开头的路径

                        * 规则:判断定义的路径是给谁用的?判断请求将来从哪儿发出
                            * 给客户端浏览器使用:需要加虚拟目录(项目的访问路径)
                                * 建议虚拟目录动态获取:request.getContextPath()
                                * <a> , <form> 重定向...
                            * 给服务器使用:不需要加虚拟目录
                                * 转发路径
                                
                        
                        

        2. 服务器输出字符数据到浏览器
            * 步骤:
                1. 获取字符输出流
                2. 输出数据

            * 注意:
                * 乱码问题:
                    1. PrintWriter pw = response.getWriter();获取的流的默认编码是ISO-8859-1
                    2. 设置该流的默认编码
                    3. 告诉浏览器响应体使用的编码

                    //简单的形式,设置编码,是在获取流之前设置
                    response.setContentType("text/html;charset=utf-8");
        3. 服务器输出字节数据到浏览器
            * 步骤:
                1. 获取字节输出流
                2. 输出数据

        4. 验证码
            1. 本质:图片
            2. 目的:防止恶意表单注册

 

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


    * 分析

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

			CREATE DATABASE day14;
			USE day14;
			CREATE TABLE USER(
			
				id INT PRIMARY KEY AUTO_INCREMENT,
				username VARCHAR(32) UNIQUE NOT NULL,
				PASSWORD VARCHAR(32) NOT NULL
			);

        3. 创建包cn.itcast.domain,创建类User

                        package cn.itcast.domain;
			/**
			 * 用户的实体类
			 */
			public class User {
			
			    private int id;
			    private String username;
			    private String password;
			
			
			    public int getId() {
			        return id;
			    }
			
			    public void setId(int 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 + '\'' +
			                '}';
			    }
			}

           4. 创建包cn.itcast.util,编写工具类JDBCUtils

                    package cn.itcast.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();
			    }
			}

                 5. 创建包cn.itcast.dao,创建类UserDao,提供login方法

			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.编写sql
			            String sql = "select * from user where username = ? and password = ?";
			            //2.调用query方法
			            User user = template.queryForObject(sql,
			                    new BeanPropertyRowMapper<User>(User.class),
			                    loginUser.getUsername(), loginUser.getPassword());
			
			
			            return user;
			        } catch (DataAccessException e) {
			            e.printStackTrace();//记录日志
			            return null;
			        }
			    }
			}

        6. 编写cn.itcast.web.servlet.LoginServlet类

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);
			
			        //4.调用UserDao的login方法
			        UserDao dao = new UserDao();
			        User user = dao.login(loginUser);
			
			        //5.判断user
			        if(user == null){
			            //登录失败
			            req.getRequestDispatcher("/failServlet").forward(req,resp);
			        }else{
			            //登录成功
			            //存储数据
			            req.setAttribute("user",user);
			            //转发
			            req.getRequestDispatcher("/successServlet").forward(req,resp);
			        }
			
			    }
			
			    @Override
			    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
			        this.doGet(req,resp);
			    }
			}

        7. 编写FailServlet和SuccessServlet类

			@WebServlet("/successServlet")
			public class SuccessServlet extends HttpServlet {
			    protected void doPost(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()+",欢迎您");
			        }
			
			
			    }		


			@WebServlet("/failServlet")
			public class FailServlet extends HttpServlet {
			    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
			        //给页面写一句话
			
			        //设置编码
			        response.setContentType("text/html;charset=utf-8");
			        //输出
			        response.getWriter().write("登录失败,用户名或密码错误");
			
			    }
			
			    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
			        this.doPost(request,response);
			    }
			}

        8. login.html中form表单的action路径的写法
            * 虚拟目录+Servlet的资源路径

        9. 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对象中

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值