Request 和 Response详解(下)

本文涉及的产品
云数据库 RDS MySQL,集群系列 2核4GB
推荐场景:
搭建个人博客
RDS MySQL Serverless 基础系列,0.5-2RCU 50GB
云数据库 RDS MySQL,高可用系列 2核4GB
简介: Request 和 Response详解

3 Respones请求重定向


  1. Response重定向(redirect):一种资源跳转方式。



(1)浏览器发送请求给服务器,服务器中对应的资源A接收到请求


(2)资源A现在无法处理该请求,就会给浏览器响应一个302的状态码+location的一个访问资源B的路径


(3)浏览器接收到响应状态码为302就会重新发送请求到location对应的访问地址去访问资源B


(4)资源B接收到请求后进行处理并最终给浏览器响应结果,这整个过程就叫重定向


  1. 重定向的实现方式:


resp.setStatus(302);
resp.setHeader("location","资源B的访问路径");


具体如何来使用,我们先来看下需求:



针对上述需求,具体的实现步骤为:


1.创建一个ResponseDemo1类,接收/resp1的请求,在doGet方法中打印resp1....


2.创建一个ResponseDemo2类,接收/resp2的请求,在doGet方法中打印resp2....


3.在ResponseDemo1的方法中使用


response.setStatus(302);


response.setHeader(“Location”,“/request-demo/resp2”) 来给前端响应结果数据


4.启动测试


(1)创建ResponseDemo1类


@WebServlet("/resp1")
public class ResponseDemo1 extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        System.out.println("resp1....");
    }
    @Override
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        this.doGet(request, response);
    }
}


(2)创建ResponseDemo2类


@WebServlet("/resp2")
public class ResponseDemo2 extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        System.out.println("resp2....");
    }
    @Override
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        this.doGet(request, response);
    }
}


(3)在ResponseDemo1的doGet方法中给前端响应数据


@WebServlet("/resp1")
public class ResponseDemo1 extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        System.out.println("resp1....");
        //重定向
        //1.设置响应状态码 302
        response.setStatus(302);
        //2. 设置响应头 Location
        response.setHeader("Location","/request-demo/resp2");
    }
    @Override
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        this.doGet(request, response);
    }
}


(4)启动测试


访问http://localhost:8080/request-demo/resp1,就可以在控制台看到如下内容:



说明/resp1/resp2都被访问到了。到这重定向就已经完成了。


虽然功能已经实现,但是从设置重定向的两行代码来看,会发现除了重定向的地址不一样,其他的内容都是一模一样,所以resposne对象给我们提供了简化的编写方式为:


resposne.sendRedirect("/request-demo/resp2")


所以第3步中的代码就可以简化为:


@WebServlet("/resp1")
public class ResponseDemo1 extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        System.out.println("resp1....");
        //重定向
        resposne.sendRedirect("/request-demo/resp2");
    }
    @Override
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        this.doGet(request, response);
    }
}


  1. 重定向的特点


  • 浏览器地址栏路径发送变化


当进行重定向访问的时候,由于是由浏览器发送的两次请求,所以地址会发生变化



  • 可以重定向到任何位置的资源(服务内容、外部均可)


因为第一次响应结果中包含了浏览器下次要跳转的路径,所以这个路径是可以任意位置资源。


  • 两次请求,不能在多个资源使用request共享数据


因为浏览器发送了两次请求,是两个不同的request对象,就无法通过request对象进行共享数据


介绍完请求重定向请求转发以后,接下来需要把这两个放在一块对比下:



以后到底用哪个,还是需要根据具体的业务来决定。


# 如果需要在资源之间传递数据,使用请求转发, 否则就用重定向


4 路径问题


  1. 问题1:转发的时候路径上没有加/request-demo而重定向加了,那么到底什么时候需要加,什么时候不需要加呢?



其实判断的依据很简单,只需要记住下面的规则即可:


  • 浏览器使用:需要加虚拟目录(项目访问路径)


  • 服务端使用:不需要加虚拟目录


对于转发来说,因为是在服务端进行的,所以不需要加虚拟目录


对于重定向来说,路径最终是由浏览器来发送请求,就需要添加虚拟目录。


掌握了这个规则,接下来就通过一些练习来强化下知识的学习:


  • <a href='路劲'>


  • <form action='路径'>


  • req.getRequestDispatcher(“路径”)


  • resp.sendRedirect(“路径”)


答案:


1.超链接,从浏览器发送,需要加
2.表单,从浏览器发送,需要加
3.转发,是从服务器内部跳转,不需要加
4.重定向,是由浏览器进行跳转,需要加。


5 Response响应字符数据


要想将字符数据写回到浏览器,我们需要两个步骤:


  • 通过Response对象获取字符输出流: PrintWriter writer = resp.getWriter();


  • 通过字符输出流写数据: writer.write(“aaa”);


接下来,我们实现通过些案例把响应字符数据给实际应用下:


  1. 返回一个简单的字符串aaa


/**
 * 响应字符数据:设置字符数据的响应体
 */
@WebServlet("/resp3")
public class ResponseDemo3 extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        response.setContentType("text/html;charset=utf-8");
        //1. 获取字符输出流
        PrintWriter writer = response.getWriter();
     writer.write("aaa");
    }
    @Override
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        this.doGet(request, response);
    }
}



  1. 返回一串html字符串,并且能被浏览器解析


PrintWriter writer = response.getWriter();
//content-type,告诉浏览器返回的数据类型是HTML类型数据,这样浏览器才会解析HTML标签
response.setHeader("content-type","text/html");
writer.write("<h1>aaa</h1>");



==注意:==一次请求响应结束后,response对象就会被销毁掉,所以不要手动关闭流。


  1. 返回一个中文的字符串你好,需要注意设置响应数据的编码为utf-8


//设置响应的数据格式及数据的编码
response.setContentType("text/html;charset=utf-8");
writer.write("你好");



6 Response响应字节数据


要想将字节数据写回到浏览器,我们需要两个步骤:


  • 通过Response对象获取字节输出流:ServletOutputStream outputStream = resp.getOutputStream();


  • 通过字节输出流写数据: outputStream.write(字节数据);


接下来,我们实现通过些案例把响应字节数据给实际应用下:


  1. 返回一个图片文件到浏览器


/**
 * 响应字节数据:设置字节数据的响应体
 */
@WebServlet("/resp4")
public class ResponseDemo4 extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        //1. 读取文件
        FileInputStream fis = new FileInputStream("d://a.jpg");
        //2. 获取response字节输出流
        ServletOutputStream os = response.getOutputStream();
        //3. 完成流的copy
        byte[] buff = new byte[1024];
        int len = 0;
        while ((len = fis.read(buff))!= -1){
            os.write(buff,0,len);
        }
        fis.close();
    }
    @Override
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        this.doGet(request, response);
    }
}



上述代码中,对于流的copy的代码还是比较复杂的,所以我们可以使用别人提供好的方法来简化代码的开发,具体的步骤是:


(1)pom.xml添加依赖


<dependency>
    <groupId>commons-io</groupId>
    <artifactId>commons-io</artifactId>
    <version>2.6</version>
</dependency>


(2)调用工具类方法


//fis:输入流
//os:输出流
IOUtils.copy(fis,os);


优化后的代码:


/**
 * 响应字节数据:设置字节数据的响应体
 */
@WebServlet("/resp4")
public class ResponseDemo4 extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        //1. 读取文件
        FileInputStream fis = new FileInputStream("d://a.jpg");
        //2. 获取response字节输出流
        ServletOutputStream os = response.getOutputStream();
        //3. 完成流的copy
        IOUtils.copy(fis,os);
        fis.close();
    }
    @Override
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        this.doGet(request, response);
    }
}


5.用户注册登录案例


接下来我们通过两个比较常见的案例,一个是注册,一个是登录来对今天学习的内容进行一个实战演练,首先来实现用户登录。


5.1 用户登录


1.登录案例分析


学习目标


  • 能够分析登录案例的步骤


  • 能够理解登录案例整体流程


内容讲解


1.开发步骤


1.画流程开发图


2.搭建项目环境


3.根据流程图编写代码


4.浏览器访问服务器,测试功能


2.画流程开发图


#web层:
    1.获取浏览器的请求数据
    2.创建实体类对象user
    3.将用户名和密码封装到对象user
    4.创建业务层对象
    5.使用业务层对象调用业务层登录方法
    6.判断业务层返回的对象是否是null
    7.是null,说明没有查到用户,登录失败,跳转到登录失败页面
    8.如果不是null,登录成功,跳转到成功页面
#service层:   
    1.定义登录方法接收web层传递的用户对象
    2.使用mybatis工具类获取session对象
    3.使用session对象调用方法获取接口Mapper对象
    4.使用Mapper对象调用接口的登录方法
    User u = mapper.login(user);
    5.返回给web层对象u
#dao层:
  1.定义接口UserMapper
  2.在接口UserMapper中定义登录方法



2.环境搭建


学习目标


  • 能够独立完成前后台环境的搭建


内容讲解


1后台环境


【1】数据库环境


create database day11_db;
use day11_db;
DROP TABLE IF EXISTS `user`;
CREATE TABLE `user` (
  `id` int(11) NOT NULL AUTO_INCREMENT,
  `username` varchar(20) NOT NULL,
  `password` varchar(32) NOT NULL,
  PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=5 DEFAULT CHARSET=utf8;
-- ----------------------------
-- Records of user
-- ----------------------------
INSERT INTO `user` VALUES ('1', 'zhangsan', '1234');
INSERT INTO `user` VALUES ('2', 'lisi', '1234');
INSERT INTO `user` VALUES ('3', 'wangwu', '1234');
INSERT INTO `user` VALUES ('4', 'zhaoliu', '1234');


【2】创建工程并创建分层包



【3】导入第三方jar包


 <dependencies>
         <!--junit-->
        <dependency>
            <groupId>junit</groupId>
            <artifactId>junit</artifactId>
            <!--<version>4.13</version>-->
            <version>4.13</version>
            <!--范围:测试存在-->
            <!--<scope>test</scope>-->
        </dependency>
        <!--mybatis核心包-->
        <dependency>
            <groupId>org.mybatis</groupId>
            <artifactId>mybatis</artifactId>
            <version>3.5.0</version>
        </dependency>
        <!--logback日志包-->
        <dependency>
            <groupId>org.slf4j</groupId>
            <artifactId>slf4j-api</artifactId>
            <version>1.7.26</version>
        </dependency>
        <dependency>
            <groupId>ch.qos.logback</groupId>
            <artifactId>logback-core</artifactId>
            <version>1.2.3</version>
        </dependency>
        <dependency>
            <groupId>ch.qos.logback</groupId>
            <artifactId>logback-classic</artifactId>
            <version>1.2.3</version>
        </dependency>
        <!--mysql驱动-->
        <dependency>
            <groupId>mysql</groupId>
            <artifactId>mysql-connector-java</artifactId>
            <version>5.1.18</version>
        </dependency>
    <dependency>
            <groupId>javax.servlet</groupId>
            <artifactId>javax.servlet-api</artifactId>
            <version>3.0.1</version>
            <!--编译 测试需要,运行时不需要-->
            <scope>provided</scope>
        </dependency>
    </dependencies>
  <!--插件管理-->
    <build>
        <plugins>
            <!--JDK编译插件-->
            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-compiler-plugin</artifactId>
                <version>3.8.1</version>
                <configuration>
                    <!-- put your configurations here -->
                    <source>1.8</source>
                    <target>1.8</target>
                    <encoding>UTF-8</encoding>
                </configuration>
            </plugin>
            <plugin>
                <groupId>org.apache.tomcat.maven</groupId>
                <artifactId>tomcat7-maven-plugin</artifactId>
                <version>2.2</version>
                <configuration>
                    <port>80</port>
                    <path>/</path>
                </configuration>
            </plugin>
        </plugins>
    </build>


【4】配置文件



db.properties:


jdbc.driver=com.mysql.jdbc.Driver
jdbc.url=jdbc:mysql://localhost:3306/day33_usermanager
jdbc.username=root
jdbc.password=1234


mybatis核心配置文件:


<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE configuration
        PUBLIC "-//mybatis.org//DTD Config 3.0//EN"
        "http://mybatis.org/dtd/mybatis-3-config.dtd">
<configuration>
    <!--加载外部的配置文件-->
    <properties resource="db.properties"></properties>
    <!--settings-->
    <settings>
        <!--开启驼峰自动映射-->
        <setting name="mapUnderscoreToCamelCase" value="true"/>
    </settings>
    <!--别名-->
    <typeAliases>
        <package name="com.itheima.sh.pojo"></package>
    </typeAliases>
    <!--mybatis环境的配置
        一个核心配置文件,可以配置多个运行环境,default默认使用哪个运行环境
    -->
    <environments default="development">
        <!--通常我们只需要配置一个就可以了, id是环境的名字 -->
        <environment id="development">
            <!--事务管理器:由JDBC来管理-->
            <!--
                事务管理器type的取值:
                1. JDBC:由JDBC进行事务的管理
                2. MANAGED:事务由容器来管理,后期学习Spring框架的时候,所有的事务由容器管理
            -->
            <transactionManager type="JDBC"/>
            <!--数据源的配置:mybatis自带的连接池-->
            <!--
                数据源:
                1. POOLED:使用mybatis创建的连接池
                2. UNPOOLED:不使用连接池,每次自己创建连接
                3. JNDI:由服务器提供连接池的资源,我们通过JNDI指定的名字去访问服务器中资源。
            -->
            <dataSource type="POOLED">
                <property name="driver" value="${jdbc.driver}"/>
                <property name="url" value="${jdbc.url}"/>
                <property name="username" value="${jdbc.username}"/>
                <property name="password" value="${jdbc.password}"/>
            </dataSource>
        </environment>
    </environments>
    <!--映射器-->
    <mappers>
        <!--加载其它的映射文件 注:注解开发是点号-->
        <!-- <package name="com.itheima.sh.dao"></package>-->
        <!--加载其它的映射文件 注:不是点号-->
        <!--<mapper resource="org/mybatis/example/BlogMapper.xml"/>-->
        <!--
            加载其它的映射文件 xml形式
                包扫描方式加载mapper映射文件,说明:
                1. 要求mapper映射文件,与mapper接口要放在同一个目录
                2. 要求mapper映射文件的名称,与mapper接口的名称要一致
            -->
        <package name="com.itheima.sh.dao"></package>
    </mappers>
</configuration>


【5】实体类


package com.itheima.sh.pojo;
public class User {
    //成员变量
    private Integer id;
    private String username;
    private String password;
    public User() {
    }
    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 + '\'' +
                '}';
    }
}


【6】工具类


package com.itheima.sh.util;
import org.apache.ibatis.io.Resources;
import org.apache.ibatis.session.SqlSession;
import org.apache.ibatis.session.SqlSessionFactory;
import org.apache.ibatis.session.SqlSessionFactoryBuilder;
import java.io.IOException;
/*
    工具类特点:
        1.位于util包
        2.私有化构造方法
        3.提供静态方法
 */
public class SqlSessionUtil {
    private static SqlSessionFactory factory = null;
    //2.私有化构造方法 目的:不能创建对象,使用类名调用本类静态方法
    private SqlSessionUtil() {
    }
    //定义静态代码块:当前类加载就执行,之后执行一次
    static {
        try {
            //1.创建工厂创造类对象
            SqlSessionFactoryBuilder builder = new SqlSessionFactoryBuilder();
            //2.获取工厂对象
            factory = builder.build(Resources.getResourceAsStream("mybatis-config.xml"));
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
    //编写静态方法获取会话对象,设置自动提交事务
    public static SqlSession getSqlSession() {
        //1.根据会话工厂对象获取会话对象
        SqlSession sqlSession = factory.openSession(true);
        //2.返回会话对象
        return sqlSession;
    }
    //编写静态方法获取会话对象,方法接收调用者传递的参数是布尔类型,决定是否设置自动提交事务
    public static SqlSession getSqlSession(boolean isAutoCommit) {
        //1.根据会话工厂对象获取会话对象
        SqlSession sqlSession = factory.openSession(isAutoCommit);
        //2.返回会话对象
        return sqlSession;
    }
    //编写静态方法接收会话对象,手动提交事务并且关闭会话
    public static void commitAndClose(SqlSession sqlSession) {
        //防止空指针异常
        if (sqlSession != null) {
            //1.手动提交事务
            sqlSession.commit();
            //2.释放资源
            sqlSession.close();
        }
    }
    //编写静态方法接收会话对象,回滚事务并且关闭会话
    public static void rollbackAndClose(SqlSession sqlSession) {
        //防止空指针异常
        if (sqlSession != null) {
            //1.回滚事务
            sqlSession.rollback();
            //2.释放资源
            sqlSession.close();
        }
    }
}


【7】UserMapper接口


package com.itheima.sh.dao;
import com.itheima.sh.pojo.User;
import org.apache.ibatis.annotations.Select;
public interface UserMapper {
    //登录
    /*
        1.username=#{username} 等号左边的username数据表的字段名  等号右边大括号中的username看此时login登录方法的形参类型
        这里login登录方法形参类型是User类型,如果形参是一个pojo属于复杂类型,看实体类User中的getXxxx()去掉get,X变为x
        例如getUsername----username,#{username}在mybatis底层调用的是User实体类中的getUsername
     */
    @Select("select * from user where username=#{username} and password=#{password}")
    User login(User user);
}


【8】测试类


package com.itheima.sh.test;
import com.itheima.sh.dao.UserMapper;
import com.itheima.sh.pojo.User;
import com.itheima.sh.utils.SessionFactoryUtils;
import org.apache.ibatis.session.SqlSession;
import org.junit.Test;
public class LoginTest {
    //查询
    @Test
    public void login(){
        //1.使用工具类获取会话对象
        SqlSession session = SessionFactoryUtils.getSession();
        //2.使用会话对象获取接口代理对象
        UserMapper mapper = session.getMapper(UserMapper.class);
        //3.使用接口代理对象调用接口中的登录方法
        User user = new User();
        user.setUsername("zhangsan");
        user.setPassword("1234");
        User u = mapper.login(user);
        //4.输出结果
        System.out.println("u = " + u);
        //5.关闭会话
        session.close();
    }
}


2前台环境


直接将资料中的内容复制到工程下面的web文件夹即可。




3.根据流程图编写代码


1.dao层代码的编写


学习目标


  • 能够编写出dao层的代码向数据库查询数据


内容讲解


【1】步骤


1.创建UserMapper接口
2.在UserMapper接口中定义登录方法
3.在登录方法上使用注解向数据库根据用户名和密码查询数据


【2】实现


package com.itheima.sh.dao;
import com.itheima.sh.pojo.User;
import org.apache.ibatis.annotations.Select;
// 1.定义接口UserMapper
public interface UserMapper {
    // 2.在接口UserMapper中定义登录方法
    //登录
    /*
        1.username=#{username} 等号左边的username数据表的字段名  等号右边大括号中的username看此时login登录方法的形参类型
        这里login登录方法形参类型是User类型,如果形参是一个pojo属于复杂类型,看实体类User中的getXxxx()去掉get,X变为x
        例如getUsername----username,#{username}在mybatis底层调用的是User实体类中的getUsername
     */
    @Select("select * from user where username=#{username} and password=#{password}")
    User login(User user);
}


内容小结


1.定义根据用户名和密码查询的方法


username=#{username}:
            1.等号左边的username看数据表的字段名
            2.#{username}:大括号中的username,由于login方法参数是User实体类类型,所以这里的username书写看User实体类中的成员变量名username
            实际上这里调用的是User实体类中的getUsername


2.service层代码的编写


学习目标


  • 能够编写业务层代码


内容讲解


【1】步骤:


  1.定义登录方法接收web层传递的用户对象
    2.使用mybatis工具类获取session对象
    3.使用session对象调用方法获取接口Mapper对象
    4.使用Mapper对象调用接口的登录方法
    User u = mapper.login(user);
    5.返回给web层对象u


【2】实现:


import org.apache.ibatis.session.SqlSession;
/*
    业务层
 */
public class UserServcie {
    //1.定义登录方法接收web层传递的用户对象
    public User login(User user) {
        //2.使用mybatis工具类获取session对象
        SqlSession session = SessionFactoryUtils.getSession();
        //3.使用session对象调用方法获取接口Mapper对象
        UserMapper mapper = session.getMapper(UserMapper.class);
        //4.使用Mapper对象调用接口的登录方法
        User u = mapper.login(user);
        //User u = mapper.login(user);
        //关闭会话对象
        session.close();
        //5.返回给web层对象u
        return u;
    }
}


内容小结


使用mybatis工具类获取接口代理对象,调用方法查询用户并返回查询的实体类对象给web层。


3.web层代码的编写


学习目标


  • 能够编写web层代码


内容讲解


【1】步骤


    1.获取浏览器的请求数据
    2.创建实体类对象user
    3.将用户名和密码封装到对象user
    4.创建业务层对象
    5.使用业务层对象调用业务层登录方法
    6.判断业务层返回的对象是否是null
    7.是null,说明没有查到用户,登录失败,跳转到登录失败页面
    8.如果不是null,登录成功,跳转到成功页面


【2】实现


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 {
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        doGet(request, response);
    }
    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        //1.获取浏览器的请求数据
        String username = request.getParameter("username");
        String password = request.getParameter("password");
        //2.创建实体类对象user
        User user = new User();
        //3.将用户名和密码封装到对象user
        user.setUsername(username);
        user.setPassword(password);
        //4.创建业务层对象
        UserServcie userServcie = new UserServcie();
        //5.使用业务层对象调用业务层登录方法
        User u = userServcie.login(user);
        //6.判断业务层返回的对象是否是null
        if (u == null) {
            //7.是null,说明没有查到用户,登录失败,跳转到登录失败页面
            request.getRequestDispatcher("/error.html").forward(request,response);
        } else {
            //8.如果不是null,登录成功,跳转到成功页面
            request.getRequestDispatcher("/success.html").forward(request,response);
        }
    }
}


5.2 用户注册


学习目标


  • 完成注册案例


内容讲解


1.开发步骤


1.画注册流程图


2.根据流程图编写代码


3.浏览器访问服务器,测试功能


2.画注册流程图


dao层:
     //1.在接口中定义根据用户名查询的方法
     //2.在接口中定义注册方法
service层:
    //1.定义注册的方法接收web层传递的user对象
    //2.获取会话对象
    //3.获取接口代理对象
    //4.使用接口代理对象调用接口中的根据用户名查询的方法
    //5.判断返回的User对象是否等于null
    //6.如果等于null,说明没有查到,可以注册,使用接口代理对象调用注册方法
    //7.提交事务
    //8.返回给web层true
    //9.如果查询的用户对象不等于null,说明用户存在,不能注册,返回给web层false
    //10.释放资源
web层:
      //1.处理请求乱码
        //2.获取页面的数据
        //3.创建User对象
        //4.将获取的数据封装到User对象中
        //5.创建业务层对象
        //6.使用业务层对象调用注册方法,将User对象作为参数传递到业务层
        //7.判断返回值
        //8.如果返回值是true,注册成功,跳转到登录页面
        //9.如果返回值是false,注册失败,跳转到失败页面


注意:注册案例需要先根据注册的用户名查询数据,如果用户名存在注册失败,用户名不存在,注册成功。



3.根据流程图编写代码


3.1dao层


import org.apache.ibatis.annotations.Insert;
import org.apache.ibatis.annotations.Param;
import org.apache.ibatis.annotations.Select;
import java.util.List;
//1.创建接口UserMapper
public interface UserMapper {
    //1.定义根据用户名查询用户的方法
    @Select("select * from user where username=#{username}")
    User queryUserByUsername(@Param("username") String username);
    //2.添加用户的方法
    @Insert("insert into user values(null,#{username},#{password})")
    void addUser(User user);
}


3.2service层


import org.apache.ibatis.session.SqlSession;
/*
    业务层
 */
public class UserService {
    //1.定义一个登录方法接收web层传递的数据
    public User login(User user) {
        //2.使用mybatis工具类调用方法获取会话对象
        SqlSession session = SessionFactoryUtils.getSession();
        //3.使用session对象调用UserMapper接口代理对象
        UserMapper mapper = session.getMapper(UserMapper.class);
        //4.使用接口代理对象mapper调用接口中的方法根据用户名和密码查询数据
        User u = mapper.login(user);
        //5.关闭session
        session.close();
        //6.返回u
        return u;
    }
    //1.定义注册方法接收web层传递的数据user
    public boolean register(User user) {
        //2.使用mybatis工具类获取session会话对象
        SqlSession session = SessionFactoryUtils.getSession();
        //3.使用会话对象调用方法获取接口UserMapper代理对象
        UserMapper mapper = session.getMapper(UserMapper.class);
        //4.根据UserMapper代理对象 调用根据用户名查询的方法
        String username = user.getUsername();
        User u = mapper.queryUserByUsername(username);
        //5.判断u是否等于null
        if (u == null) {
            //6.没有查到用户,可以注册,直接调用添加方法
            mapper.addUser(user);
            //提交事务
            session.commit();
            //7.关闭session
            session.close();
            //8.返回true
            return true;
        } else {
            //9.注册失败,关闭session
            session.close();
            //10.返回false
            return false;
        }
    }
}


小结:


1.在业务层我们主要是先根据用户名查询用户是否存在,如果不存在在执行添加用户方法,存在则不添加。


2.别忘记提交事务,增删改需要提交事务


3.3web层


import org.apache.commons.beanutils.BeanUtils;
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;
import java.lang.reflect.InvocationTargetException;
@WebServlet("/registerServlet")
public class RegisterServlet extends HttpServlet {
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        doGet(request, response);
    }
    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        //1.处理post请求乱码
        request.setCharacterEncoding("utf-8");
        //2.接收请求数据封装到User实体类对象中
        User user = new User();
        try {
            BeanUtils.populate(user, request.getParameterMap());
            //3. 创建业务层对象
            UserService service = new UserService();
            //4. 使用业务层对象调用注册方法将用户数据传递到service层
            boolean result = service.register(user);
            //5. 判断result结果
            if (result) {
                //6. 表示注册成功, 跳转到登录页面
                response.sendRedirect("/login.html");
            } else {
                //7. 注册失败,跳转到失败页面
                response.sendRedirect("/registerError.html");
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}


3.4页面实现


register.html页面:


<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>欢迎注册</title>
    <link href="css/register.css" rel="stylesheet">
</head>
<body>
<div class="form-div">
    <div class="reg-content">
        <h1>欢迎注册</h1>
        <span>已有帐号?</span> <a href="login.html">登录</a>
    </div>
    <form id="reg-form" action="#" method="get">
        <table>
            <tr>
                <td>用户名</td>
                <td class="inputs">
                    <input name="username" type="text" id="username">
                    <br>
                    <span id="username_err" class="err_msg" style="display: none">用户名不太受欢迎</span>
                </td>
            </tr>
            <tr>
                <td>密码</td>
                <td class="inputs">
                    <input name="password" type="password" id="password">
                    <br>
                    <span id="password_err" class="err_msg" style="display: none">密码格式有误</span>
                </td>
            </tr>
        </table>
        <div class="buttons">
            <input value="注 册" type="submit" id="reg_btn">
        </div>
        <br class="clear">
    </form>
</div>
</body>
</html>


registerError.html


<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
</head>
<body>
    <h2>注册失败,<a href="register.html">重新注册</a></h2>
</body>
</html>
相关实践学习
如何在云端创建MySQL数据库
开始实验后,系统会自动创建一台自建MySQL的 源数据库 ECS 实例和一台 目标数据库 RDS。
全面了解阿里云能为你做什么
阿里云在全球各地部署高效节能的绿色数据中心,利用清洁计算为万物互联的新世界提供源源不断的能源动力,目前开服的区域包括中国(华北、华东、华南、香港)、新加坡、美国(美东、美西)、欧洲、中东、澳大利亚、日本。目前阿里云的产品涵盖弹性计算、数据库、存储与CDN、分析与搜索、云通信、网络、管理与监控、应用服务、互联网中间件、移动服务、视频服务等。通过本课程,来了解阿里云能够为你的业务带来哪些帮助 &nbsp; &nbsp; 相关的阿里云产品:云服务器ECS 云服务器 ECS(Elastic Compute Service)是一种弹性可伸缩的计算服务,助您降低 IT 成本,提升运维效率,使您更专注于核心业务创新。产品详情: https://www.aliyun.com/product/ecs
相关文章
|
5月前
|
应用服务中间件 数据安全/隐私保护
|
5月前
|
Java 应用服务中间件 API
Request&Response(1)
Request&Response
59 0
|
5月前
|
存储 Java 应用服务中间件
Request&Response(3)
Request&Response
53 0
|
5月前
|
Java Maven 数据安全/隐私保护
Request&Response(2)
Request&Response
45 0
|
5月前
|
前端开发 Java 应用服务中间件
Request&Response(4)
Request&Response
38 0
|
5月前
|
Java 数据库连接 数据库
Request&Response(5)
Request&Response
47 0
|
6月前
|
Java 数据库连接 数据库
Request 和 Response详解(下)
Request 和 Response详解(下)
74 1
|
6月前
|
存储 缓存 前端开发
Request 和 Response详解(中)
Request 和 Response详解(中)
79 0
|
6月前
|
前端开发 Java 应用服务中间件
Request 和 Response详解(上)
Request 和 Response详解(上)
119 0
|
存储
Request和Response详解
Request和Response详解
110 0