《黑马旅游网》综合案例day01

《黑马旅游网》综合案例day01

  1. 前言
    为了巩固web基础知识,提升综合运用能力,故而讲解此案例。
    要求,每位同学能够独立完成此案例。

  2. 项目导入
    2.1 点击+按钮
    在这里插入图片描述
    2.2 选择travel项目的pom.xml文件,点击ok,完成项目导入。需要等待一小会,项目初始化完成。
    在这里插入图片描述

  3. 启动项目
    3.1 方式一:
    在这里插入图片描述
    3.2方式二:配置maven快捷启动
    在这里插入图片描述在这里插入图片描述

  4. 技术选型

    4.1Web层
    a)Servlet:前端控制器
    b)html:视图
    c)Filter:过滤器
    d)BeanUtils:数据封装
    e)Jackson:json序列化工具
    4.2Service层
    f)Javamail:java发送邮件工具
    g)Redis:nosql内存数据库
    h)Jedis:java的redis客户端
    4.3Dao层
    i)Mysql:数据库
    j)Druid:数据库连接池
    k)JdbcTemplate:jdbc的工具

  5. 创建数据库

    -- 创建数据库
    CREATE DATABASE travel;
    -- 使用数据库
    USE travel;
    --创建表
    复制提供好的travel.sql
    
  6. 注册功能

    6.1页面效果:register.html
    在这里插入图片描述

    6.2功能分析
    在这里插入图片描述
    6.3代码实现

    6.3.1前台代码实现

    6.3.2表单校验

    提升用户体验,并减轻服务器压力。

    <script>
    	/*
    	表单校验:
    		1.用户名:单词字符,长度8-20位
    		2.密码:单词字符,长度8到20位
    		3.email:邮件格式
    		4.姓名:非空
    		5.手机号:手机号格式
    		6.出生日期:非空
    		7.验证码:非空
    	*/
    	//校验用户名
    	//单词字符,长度8到20位
    	function  checkUsername(){
          
          
    		//1.获取用户名
    		var username = $("#username").val();
    		//2.定义正则
    		var reg_username = /^\w{8,20}$/;
    		//3.判断,给出提示信息
    		var flag = reg_username.test(username);
    		if (flag){
          
          
    			//用户名合法
    			$("#username").css("border","");
    		}else{
          
          
    			//用户名不合法,加一个红色边框
    			$("#username").css("border","1px solid red");
    		}
    		
    		return flag;
    	}
    
    	//校验密码
    	function checkPassword(){
          
          
    		//1.获取用户名
    		var password = $("#password").val();
    		//2.定义正则
    		var reg_password = /^\w{8,20}$/;
    		//3.判断,给出提示信息
    		var flag = reg_password.test(password);
    		if (flag){
          
          
    			//用户名合法
    			$("#password").css("border","");
    		}else{
          
          
    			//用户名不合法,加一个红色边框
    			$("#password").css("border","1px solid red");
    		}
    
    		return flag;
    	}
    
    	//校验邮箱
    	function checkEmail(){
          
          
    		//1.获取邮箱
    		var email = $("#email").val();
    		//2.定义正则 [email protected]
    		var reg_email = /^\w+@\w+\.\w+$/;
    		//3.判断
    		var flag = reg_email.test(email);
    		if (flag){
          
          
    			$("#email").css("border","");
    		}else{
          
          
    			$("#email").css("border","1px solid red");
    		}
    
    		return flag;
    	}
    
    	$(function () {
          
          
             //当表单提交时,调用所有的校验方法
    		$("#registerForm").submit(function(){
          
          
    			return checkUsername() && checkPassword() && checkEmail();
                 //如果这个方法没有返回值,或者返回为true,则表单提交,如果返回为false,则表单不提交
    		});
    
             //当某一个组件失去焦点是,调用对应的校验方法
    		$("#username").blur(checkUsername);
            $("#password").blur(checkPassword);
            $("#email").blur(checkEmail);
         });  
    </script>
    

    6.3.3异步(ajax)提交表单

    在此使用异步提交表单是为了获取服务器响应的数据。

    因为我们前台使用的是html作为视图层,不能够直接从servlet相关的域对象获取值,只能通过ajax获取响应数据

    //当表单提交时,调用所有的校验方法
    $("#registerForm").submit(function(){
          
          
    	//1.发送数据到服务器
    	if (checkUsername() && checkPassword() && checkEmail()){
          
          
    		//校验通过,发送ajax请求,提交表单数据  username=zhangsan&password=123
    		$.post("registUserServlet",$(this).serialize(),function(data){
          
          
    			//处理服务器的数据
    		});
    	}
    	//2.跳转页面,e不让页面跳转
    	return false;
    	//如果这个方法function()没有返回值,或者返回值为true,则表单提交,如果返回值为false,则表单不提交
    });
    

    6.3.4后台代码实现

    6.3.5编写RegistUserServlet

    @WebServlet("/registUserServlet")
    public class RegistUserServlet extends HttpServlet {
          
          
        protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
          
          
            //验证校验
            String check = request.getParameter("check");
            //从sesion中获取验证码
            HttpSession session = request.getSession();
            String checkcode_server = (String) session.getAttribute("CHECKCODE_SERVER");
            session.removeAttribute("CHECKCODE_SERVER");//为了保证验证码只能使用一次
            //比较
            if(checkcode_server == null || !checkcode_server.equalsIgnoreCase(check)){
          
          
                //验证码错误
                ResultInfo info = new ResultInfo();
                //注册失败
                info.setFlag(false);
                info.setErrorMsg("验证码错误");
                //将info对象序列化为json
                ObjectMapper mapper = new ObjectMapper();
                String json = mapper.writeValueAsString(info);
                response.setContentType("application/json;charset=utf-8");
                response.getWriter().write(json);
                return;
            }
    
            //1.获取数据
            Map<String, String[]> map = request.getParameterMap();
            
            //2.封装对象
            User user = new User();
            try {
          
          
                BeanUtils.populate(user,map);
            } catch (IllegalAccessException e) {
          
          
                e.printStackTrace();
            } catch (InvocationTargetException e) {
          
          
                e.printStackTrace();
            }
            
            //3.调用service完成注册
            UserService service = new UserServiceImpl();
            boolean flag = service.regist(user);
            ResultInfo info = new ResultInfo();
            //4.响应结果
            if(flag){
          
          
                //注册成功
                info.setFlag(true);
            }else{
          
          
                //注册失败
                info.setFlag(false);
                info.setErrorMsg("注册失败!");
            }
    
            //将info对象序列化为json
            ObjectMapper mapper = new ObjectMapper();
            String json = mapper.writeValueAsString(info);
    
            //将json数据写回客户端
            //设置content-type
            response.setContentType("application/json;charset=utf-8");
            response.getWriter().write(json);
        }
    
        protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
          
          
            this.doPost(request, response);
        }
    }
    

    6.3.6编写UserService以及UserServiceImpl

    public class UserServiceImpl implements UserService {
          
          
        private UserDao userDao = new UserDaoImpl();
        /**
         * 注册用户
         * @param user
         * @return
         */
        @Override
        public boolean regist(User user) {
          
          
            //1.根据用户名查询用户对象
            User u = userDao.findByUsername(user.getUsername());
            //判断u是否为null
            if(u != null){
          
          
                //用户名存在,注册失败
                return false;
            }
            //2.保存用户信息
            userDao.save(user);
            return true;
        }
    }
    

    6.3.7编写UserDao以及UserDaoImpl

    public class UserDaoImpl implements UserDao {
          
          
    
        private JdbcTemplate template = new JdbcTemplate(JDBCUtils.getDataSource());
    
        @Override
        public User findByUsername(String username) {
          
          
            User user = null;
            try {
          
          
                //1.定义sql
                String sql = "select * from tab_user where username = ?";
                //2.执行sql
                user = template.queryForObject(sql, new BeanPropertyRowMapper<User>(User.class), username);
            } catch (Exception e) {
          
          
    
            }
            return user;
        }
    
        @Override
        public void save(User user) {
          
          
            //1.定义sql
            String sql = "insert into tab_user(username,password,name,birthday,sex,telephone,email) values(?,?,?,?,?,?,?)";
            //2.执行sql
            template.update(sql,user.getUsername(),
                    user.getPassword(),
                    user.getName(),
                    user.getBirthday(),
                    user.getSex(),
                    user.getTelephone(),
                    user.getEmail());
        }
    }
    

    6.3.8邮件激活

    为什么要进行邮件激活?

    为了保证用户填写的邮箱是正确的。将来可以推广一些宣传信息,到用户邮箱中。

    6.3.9 发送邮件

     1.申请邮箱
     
     2.开启授权码
     
     3.在MailUtils中设置自己的邮箱账号和密码(授权码)
     
     邮件工具类:MailUtils,调用其中sendMail方法可以完成邮件发送
    

    6.3.10 用户点击邮件激活

    经过分析,发现,用户激活其实就是修改用户表中的status为‘Y’

    分析:
    在这里插入图片描述
    发送邮件代码:

    @Override
    public boolean regist(User user) {
          
          
        //1.根据用户名查询用户对象
        User u = userDao.findByUsername(user.getUsername());
        //判断u是否为null
        if (u != null) {
          
          
            //用户名存在,注册失败
            return false;
        }
        //2.保存用户信息
        //2.1设置激活码,唯一字符串
        user.setCode(UuidUtil.getUuid());
        //2.2设置激活状态
        user.setStatus("N");
        userDao.save(user);
    
        //3.激活邮件发送,邮件正文?
        String content = "<a href='http://localhost/travel/activeUserServlet?code="+user.getCode()+"'+user.get>点击激活【黑马旅游网】</a>";
        MailUtils.sendMail(user.getEmail(),content,"激活邮件");
        return true;
    }
    

    修改保存Dao代码,加上存储statuscode 的代码逻辑

    @Override
    public void save(User user) {
          
          
        //1.定义sql
        String sql = "insert into tab_user(username,password,name,birthday,sex,telephone,email,status,code) values(?,?,?,?,?,?,?,?,?)";
        //2.执行sql
        template.update(sql,user.getUsername(),
                user.getPassword(),
                user.getName(),
                user.getBirthday(),
                user.getSex(),
                user.getTelephone(),
                user.getEmail(),
                user.getStatus(),
                user.getCode()
        );
    }
    

    激活代码实现:ActiveUserServlet

    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
          
          
        //1.获取激活码
        String code = request.getParameter("code");
        if (code != null) {
          
          
            //2.调用service完成激活
            UserService service = new UserServiceImpl();
            boolean flag = service.active(code);
    
            //3.判断标记
            String msg = null;
            if (flag) {
          
          
                //激活成功
                msg = "激活成功,请<a href='login.html'>登录</a>";
            } else {
          
          
                //激活失败
                msg = "激活失败,请联系管理员!";
            }
            response.setContentType("text/html;charset=utf-8");
            response.getWriter().write(msg);
        }
    }
    

    UserService:active

    @Override
    public boolean active(String code) {
          
          
        //1.根据激活码查询用户对象
        User user = userDao.findByCode(code);
        if(user != null){
          
          
            //2.调用dao的修改激活状态的方法
            userDao.updateStatus(user);
            return true;
        }else{
          
          
            return false;
        }
    
    }
    

    UserDao:findByCode updateStatus

    /**
     * 根据激活码查询用户对象
     * @param code
     * @return
     */
    @Override
    public User findByCode(String code) {
          
          
        User user = null;
        try {
          
          
            String sql = "select * from tab_user where code = ?";
    
            user = template.queryForObject(sql,new BeanPropertyRowMapper<User>(User.class),code);
        } catch (DataAccessException e) {
          
          
            e.printStackTrace();
        }
    
        return user;
    }
    
    /**
     * 修改指定用户激活状态
     * @param user
     */
    @Override
    public void updateStatus(User user) {
          
          
        String sql = " update tab_user set status = 'Y' where uid=?";
        template.update(sql,user.getUid());
    }
    
  7. 登录

    7.1分析
    在这里插入图片描述
    7.2代码实现

    7.2.1前台代码

    login.html

    <script>
    	$(function(){
          
          
    		//1.给登录按钮绑定单击事件
    		$("#btn_sub").click(function(){
          
          
    			//2.发送ajax请求,提交表单数据
    			$.post("loginServlet",$("#loginForm").serialize(),function(data){
          
          
    				//data:{flag:false,errorMsg:''}
    				if(data.flag){
          
          
    					//登录成功
    					location.href="index.html";
    				}else{
          
          
    					//登陆失败
    					$("#errorMsg").html(data.errorMsg);
    				}
    			})
    		})
    	});
    </script>
    

    7.2.2后台代码

    LoginServlet

    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
          
          
        //1.获取用户名和密码数据
        Map<String, String[]> map = request.getParameterMap();
        //2.封装User对象
        User user = new User();
        try {
          
          
            BeanUtils.populate(user,map);
        } catch (IllegalAccessException e) {
          
          
            e.printStackTrace();
        } catch (InvocationTargetException e) {
          
          
            e.printStackTrace();
        }
    
        //3.调用Service查询
        UserService service = new UserServiceImpl();
        User u  = service.login(user);
    
        ResultInfo info = new ResultInfo();
    
        //4.判断用户对象是否为null
        if(u == null){
          
          
            //用户名密码或错误
            info.setFlag(false);
            info.setErrorMsg("用户名密码或错误");
        }
        //5.判断用户是否激活
        if(u != null && !"Y".equals(u.getStatus())){
          
          
            //用户尚未激活
            info.setFlag(false);
            info.setErrorMsg("您尚未激活,请激活");
        }
        //6.判断登录成功
        if(u != null && "Y".equals(u.getStatus())){
          
          
            request.getSession().setAttribute("user",u);//登录成功标记
            info.setFlag(true);
        }
    
        //响应数据
        ObjectMapper mapper = new ObjectMapper();
    
        response.setContentType("application/json;charset=utf-8");
        mapper.writeValue(response.getOutputStream(),info);
    
    }
    

    UserService

    public User login(User user) {
          
          
        return userDao.findByUsernameAndPassword(user.getUsername(),user.getPassword());
    }
    

    UserDao

    @Override
    public User findByUsernameAndPassword(String username, String password) {
          
          
       User user = null;
       try {
          
          
           //1.定义sql
           String sql = "select * from tab_user where username = ? and password = ?";
           //2.执行sql
           user = template.queryForObject(sql, new BeanPropertyRowMapper<User>(User.class), username,password);
       } catch (Exception e) {
          
          
    
       }
        return user;
    }
    

    7.2.3index页面中用户姓名的提示信息功能

    header.html

    <script>
        $(function(){
          
          
            $.get("findUserServlet",{
          
          },function(data){
          
          
                //{uid:1,name:'李四'}
                var msg="欢迎回来,"+data.name;
                $("#span_username").html(msg);
            });
        });
    </script>
    

    findUserServlet代码

    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
          
          
        //从session中获取登录用户
        Object user = request.getSession().getAttribute("user");
    
        //将user写回客户端
        ObjectMapper mapper = new ObjectMapper();
        response.setContentType("application/json;charset=utf-8");
        mapper.writeValue(response.getOutputStream(),user);
    }
    
  8. 退出

    什么叫做登录了?session中有user对象。

    实现步骤:

     1.访问servlet,将session销毁
     
     2.跳转到登录页面
    

    代码实现:

    Header.html

    <div class="login">
    	<span id="span_username"></span>
    	<a href="myfavorite.html" class="collection">我的收藏</a>
    	<a href="javascript:location.href='exieServlet';">退出</a>
    </div>
    

    exieServlet:

    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
          
          
    	//1.销毁session
    	request.getSession().invalidate();
    	System.out.println("exitServlet");
    	//2.跳转登录页面
    	response.sendRedirect(request.getContextPath()+"/login.html");
    }
    

踩坑:

  1. 数据库JDBCUtils工具类中,druid.properties
    在这里插入图片描述
  2. 收不到测试邮件
    若发件人设的是QQ号,则先将QQ邮箱的POP3/SMTP服务开启,再将PASSWORD的值设为获得的授权码,
    在这里插入图片描述
  3. 激活邮件,无法点击,鼠标右键,几个选项都可以打开
    在这里插入图片描述
    《黑马旅游网》综合案例day02

猜你喜欢

转载自blog.csdn.net/weixin_44505194/article/details/105604911