笔记-JavaWeb学习之旅13

时间:2022-12-28 13:02:47
验证码案列

昨天晚上出现的500错误原因在于验证码没有获取到,获取验证码是应该获取的是共享域中的验证码,而我把获取值得键给写成了jsp中的键,而不是内存生成图片中,然后把图片上传到共享域中的键。这两个键搞混了,所以获取不到验证码。

<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
    <title>login</title>


    <script>
        window.onload = function(){
            document.getElementById("img").onclick = function(){
                this.src="/day14/checkCodeDemo4?time="+new Date().getTime();
            }
        }


    </script>
    <style>
        div{
            color: red;
        }

    </style>
</head>
<body>

<form action="/day14/loginServlet" method="post">
    <table>
        <tr>
            <td>用户名</td>
            <td><input type="text" name="username"></td>
        </tr>
        <tr>
            <td>密码</td>
            <td><input type="password" name="password"></td>
        </tr>
        <tr>
            <td>验证码</td>
            <td><input type="text" name="checkCode"></td>
        </tr>
        <tr>
            <td colspan="2"><img id="img" src="/day14/checkCodeDemo4"></td>
        </tr>
        <tr>
            <td colspan="2"><input type="submit" value="登录"></td>
        </tr>
    </table>


</form>


<div><%=request.getAttribute("cc_error") == null ? "" : request.getAttribute("cc_error")%></div>
<div><%=request.getAttribute("login_error") == null ? "" : request.getAttribute("login_error") %></div>

</body>
</html>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
    <title>Title</title>
</head>
<body>

<h1><%=request.getSession().getAttribute("user")%>,欢迎您</h1>

</body>
</html>
package com.data;

import javax.imageio.ImageIO;
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.awt.*;
import java.awt.image.BufferedImage;
import java.io.IOException;
import java.util.Random;

@WebServlet("/checkCodeDemo4")
public class checkCode extends HttpServlet {
    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        this.doGet(req,resp);
    }

    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        int width = 100;
        int height = 50;
        //创建对象,该对象可以在内存中生成图片
        //BufferedImage是Image的一个子类,Image和BufferedImage的主要作用就是将一副图片加载到内存中。
        BufferedImage image = new BufferedImage(width, height, BufferedImage.TYPE_INT_RGB);//这个类第一次接触,好陌生,好抽象
        //图片加载进内存了,就尽情的虐待他吧
        //美化图片
        //填充背景色
        Graphics g = image.getGraphics();
        g.setColor(Color.PINK);
        g.fillRect(0,0,width,height);
        //画边框
        g.setColor(Color.BLUE);
        g.drawRect(0,0,width-1,height-1);
        String str="ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789";

        //  写验证码
        Random ran = new Random();
        StringBuilder sb = new StringBuilder();//初始容量为16字节
        //随机生成4个字符
        for(int i = 1;i<=4;i++){
            int index = ran.nextInt(str.length());
            char ch = str.charAt(index);
            sb.append(ch);//这个方法可以接收任意类型的数据
            g.drawString(ch+"",width/5*i,height/2);//ch代表要绘制的字符,x,y代表绘制的坐标
        }
        String value_checkCode_session = sb.toString();//把字符变成字符串
        System.out.println(value_checkCode_session);
        //创建对象,使用方法将验证码存入session
        req.getSession().setAttribute("key_checkCode_session",value_checkCode_session);//把数据放入共享域
        g.setColor(Color.GREEN);//线条的颜色
        //画10条干扰线
        for (int i = 0 ; i< 10 ;i++){
            int x1 = ran.nextInt(width);
            int x2 = ran.nextInt(width);

            int y1 = ran.nextInt(height);
            int y2 = ran.nextInt(height);
            g.drawLine(x1,y1,x2,y2);//参数是坐标
        }


        //将图片输出到页面上
        ImageIO.write(image,"jpg",resp.getOutputStream());

    }
}
package com.data;

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 javax.servlet.http.HttpSession;
import java.io.IOException;


@WebServlet("/loginServlet")
public class LoginServlet extends HttpServlet {
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        request.setCharacterEncoding("utf-8");
        //根据键获取值
        String username = request.getParameter("username");
        String password = request.getParameter("password");
        String checkCode = request.getParameter("checkCode");//这里的checkCode是用户输入的验证码
        //要先看看获取的验证码是否正确
        //先得取得内存中生成的验证码,
        HttpSession session = request.getSession();//创建HttpSession对象
        String checkCode1 = (String)session.getAttribute("key_checkCode_session");//这里的checkCode是内存中获得的验证码
        //把获得的验证码删除,确保只使用一次
        session.removeAttribute("checkCode_session");
        //判断验证码是否正确,忽略大小写
        if(checkCode1!=null&&checkCode1.equalsIgnoreCase(checkCode)){
            //验证码正确
            //判读用户名和密码是否一致
            if("zhangsan".equals(username)&&"123".equals(password)){
                //存储信息,
                session.setAttribute("user",username);
                //重定向到success.jsp页面
                response.sendRedirect(request.getContextPath()+"/success.jsp");

            }else{
                //储存信息到request
                request.setAttribute("login_error","用户名或密码错误");
                //转发都登录页面
                request.getRequestDispatcher("/login.jsp").forward(request,response);
            }
        } else{
            //验证码不一致
            //存储提示信息到request
            request.setAttribute("cc_error","验证码错误,请重试");
            //转发到登录页面
            request.getRequestDispatcher("/login.jsp").forward(request,response);
        }

    }

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

1.指令

作用:用于配置JSP页面,导入资源文件

格式:<%@ 指令名称 属性名1=属性值1 属性名2=属性值2 ...%>

指令分类:

1.page :配置JSP页面的

  • contentType:等同于response.setContentType(),设置响应体的mime类型
  • import:导包
  • errorPage:当前页面发生异常后,会自动跳转到指定的错误页面
  • isErrorPage:标识当前是否是错误页面,true,是,可以使用内置对象exception,false:否,默认值,不可以设置内置对象exception

2.include:页面包含的,导入页面的资源文件

  • <% @include file="要导入的资源文件名称"%>

3.taglib :导入资源

  • <%@ taglib prefix="c" uri="文件的地址" %>
  • prefix :前缀,自定义的

2.注释:

  • html注释: 只能注释html代码片段
  • jsp注释:<%-- --%>可以注释所有
3.内置对象

在jsp页面中不需要创建,直接使用的对象,一共有9个

变量名 真实类型 作用
pageContext PageContext 当前页面共享数据,还可以获取其他八个内置对象
request HttpServletRequest 一次请求访问的多个资源
session HttpSession 一次会话的多个请求间
application ServletContext 所有用户间共享数据
response HttpServletResponse 响应对象
page Object 当前页面的对象
out jspWriter 输出对象,数据输出到页面上
config ServletConfig Servlet的配置对象
exception Throwable 异常对象
MVC:开发模式 模型视图控制器

M:Model,模型。JavaBean,完成具体的业务操作,如,查询数据库,封装对象

V:view,视图,JSP,展示数据

C:Controller,控制器。Servlet,获取用户输入,调用模型,将数据交给视图进行展示

EL表达式:

1.概念:Expression Language 表达式语言

2.作用:替换和简化jsp页面中java代码的编写

3.语法:${表达式}

4.注意:jsp默认支持el表达式,浏览器会解析el表达式,如果让浏览器忽略jsp页面中所有的el表达式可以设置jsp中page指令中:isELIgnored="true"会 忽略当前jsp页面中所有的el表达式,或者使用${表达式}:会忽略当前这个el表达式。

EL_表达式获取域中的值

1.el表达式只能从域对象中获取值

2.语法:

  1. ${域名称.键名}:从指定域中获取指定键的值
    • 有四种域
    • 1.pageScope 2.requestScope 3.sessionScope 4.applicationScope
  2. ${键名}:表示依次从最小的域中查找是否有该键对应的值,知直到找到为止,如果域有多个同名的键,则从小范围的域开始找起。
获取对象,List集合,Map集合的值

对象格式:${域名称.键名.属性名}

List集合格式:${域名称.键名[索引]}

Map集合格式:${域名称.键名.key名称}或者${域名称.键名["key名称"]}

package com.data.jsp;
import java.text.SimpleDateFormat;
import java.util.Date;

public class User {
    private String name;
    private int age;
    private Date birthday;
    //逻辑视图
    //需要显示年月日的日期格式需要创建一个方法
    public String getBirStr(){
        //格式化日期
        if(birthday !=null){
            SimpleDateFormat sdf = new SimpleDateFormat("yyyy年MM月dd日 HH:mm:ss");
            //返回字符串
            return sdf.format(birthday);
        }else{
            return "";
        }
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }

    public Date getBirthday() {
        return birthday;
    }

    public void setBirthday(Date birthday) {
        this.birthday = birthday;
    }
}
<%@ page import="com.data.jsp.User" %>
<%@ page import="java.util.*" %><%--
  Created by IntelliJ IDEA.
  User: Yuan
  Date: 2019/6/11
  Time: 18:20
  To change this template use File | Settings | File Templates.
--%>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
    <title>Title</title>
</head>
<body>
    <%--给User设置值--%>
    <%
        User user = new User();
        user.setName("张三");
        user.setAge(23);
        user.setBirthday(new Date());
//        共享域
        request.setAttribute("u",user);
        //创建List集合
        List list = new ArrayList();
        list.add("云想衣裳花想容");
        list.add("春风扶槛露华浓");
//        添加对象进去看看
        list.add(user);
        request.setAttribute("list",list);

        //创建Map集合
        Map map = new HashMap();
        map.put("sname","李四");
        map.put("gender","男");
        //添加用户进去看看
        map.put("user",user);
        request.setAttribute("map",map);
    %>
<h3>el获取对象中的值</h3>
    <%--获取对象的值需要通过对象的方法来获取--%>
        <%--setter或getter方法,去掉set或get,在将剩余部分,首字母变小写--%>
${requestScope.u.name}<br>
${requestScope.u.age}<br>
${requestScope.u.birthday}<br>
${requestScope.u.birStr}<br>
<h3>el获取List集合中的值</h3>
${requestScope.list}<br><%--获取的是整个集合--%>
${requestScope.list[0]}<br>
${requestScope.list[1]}<br>
${requestScope.list[2].name}<br>
${requestScope.list[2].age}<br>
<h3>el获取Map集合中的值</h3>
${requestScope.map}<br>
${requestScope.map.sname}<br>
${requestScope.map.gender}<br>
${requestScope.map.user.name}<br>
</body>
</html>

笔记-JavaWeb学习之旅13

空运算符:empty

功能:用于判断字符串,集合,数组对象是否为null并且长度是否为0

格式:{empty list}

3.隐式对象:

el表达式中有11个隐式对象

pageContext:获取jsp其他八个内置对象

${pageContext.request.contextPath}:动态获取虚拟目录