JavaWeb学习笔记(二)

目录

1、Jsp

1.1、什么是 jsp,它有什么用?

1.2、jsp如何访问

1.3、jsp的本质

1.4、jsp的三种语法

1.4.1、jsp头部的page指令

1.4.2、jsp中的常用脚本

        1.4.2.1、声明脚本(极少使用)

        1.4.2.2、表达式脚本

        1.4.2.3、代码脚本

 1.4.3、jsp中的三种注释

1.5、jsp九大内置对象

1.6、jsp四大域对象

1.7、jsp中的out 输出和 response.getWriter 输出的区别

 1.8、jsp的常用标签

1.8.1、jsp静态包含

1.8.2、jsp动态包含

1.8.3、jsp 标签-转发

 1.9、ServletContextListener 监听器

2、EL表达式

2.1、什么是 EL 表达式,EL 表达式的作用

2.2、EL 表达式搜索域数据的顺序

2.3、EL表达式输出 Bean 的普通属性,数组属性。List 集 合属性,map 集合属性

2.4、EL 表达式——运算

2.4.1、关系运算

2.4.2、逻辑运算 

 2.4.3、算数运算

 2.4.4、empty运算

2.4.5、三元运算

2.4.6、" . "运算和 [ ] 中括号运算符

2.4.7、EL 表达式的 11 个隐含对象

        2.4.7.1、EL 获取四个特定域中的属性

        2.4.7.2、pageContext 对象的使用

        2.4.7.3、EL 表达式其他隐含对象的使用

3、JSTL标签库

3.1、JSTL 标签库的使用步骤

3.2、core 核心库使用

3.2.0、使用过程遇到的问题

3.2.1、(使用很少)

3.2.2、

3.2.3、 标签

3.2.4、

4、文件的上传和下载

4.0、使用过程遇到的坑

4.1、文件的上传介绍

4.1.1、文件上传、HTTP协议说明

 4.1.2、commons-fileupload.jar 常用 API 介绍说明

4.1.3、文件上传的实现

4.2、文件上传

4.2.1、实现(中文乱码)

4.2.2、中文乱码解决----URLEncoder编码(IE和谷歌浏览器)

4.2.3、中文乱码解决------BASE64编码(火狐)

4.2.4、最终版

5、MVC概念

 6、cookie

6.1、什么是Cookie

6.2、如何创建Cookie

6.3、服务器如何获取Cookie

6.4、Cookie值的修改

6.4.1、方案一: 

6.4.2、方案二

6.5、浏览器查看Cookie

6.6、Cookie生命控制 

6.7、Cookie有效路径Path的设置

6.8、Cookie练习--免输入用户名登录

7、Session会话

7.1、什么是Session会话

7.2、如何创建Session和获取(id号,是否为新)

7.3、Session域的获取

7.4、Session生命周期控制

7.5、浏览器和 Session 之间关联的技术内幕


1、Jsp

1.1、什么是 jsp,它有什么用?

jsp 的全换是 java server pages。Java 的服务器页面。
jsp 的主要作用是代替 Servlet 程序回传 html 页面的数据。 
因为 Servlet 程序回传 html 页面数据是一件非常繁锁的事情。开发成本和维护成本都极高。

Servlet 回传 html 页面数据的代码:

package Servlet;

import com.sun.net.httpserver.HttpsServer;
import jakarta.servlet.ServletException;
import jakarta.servlet.http.HttpServlet;
import jakarta.servlet.http.HttpServletRequest;
import jakarta.servlet.http.HttpServletResponse;

import java.io.IOException;
import java.io.PrintWriter;

/**
 * @Author zhang
 * @Date 2021/12/23 13:08
 * @Version 1.0
 */
public class PrintHtml extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        //通过响应的回传流回传html页面数据
        PrintWriter writer = resp.getWriter();
        resp.setContentType("text/html;character=UTF-8");
        writer.write("<!DOCTYPE html>\n" +
                "<html lang=\"en\">\n" +
                "<head>\n" +
                "    <meta charset=\"UTF-8\">\n" +
                "    <title>Title</title>\n" +
                "</head>\n" +
                "<body>\n" +
                "    这是HTML页面数据\n" +
                "</body>\n" +
                "</html>");
    }
}

1.2、jsp如何访问

jsp 页面和 html 页面一样,都是存放在 web 目录下。访问也跟访问 html 页面一样。 
比如:
    在 web 目录下有如下的文件: 
    web 目录
        a.html 页面 访问地址是 =======>>>>>> http://ip:port/工程路径/a.html 
        b.jsp 页面 访问地址是 =======>>>>>> http://ip:port/工程路径/b.jsp

1.3、jsp的本质

jsp 页面本质上是一个 Servlet 程序。 
当我们第一次访问 jsp 页面的时候。Tomcat 服务器会帮我们把 jsp 页面翻译成为一个 java 源文件。并且对它进行编译成 为.class 字节码程序。我们打开 java 源文件不难发现其里面的内容的开头是:

跟踪原代码发现,HttpJspBase 类。它直接地继承了 HttpServlet 类。也就是说。jsp 翻译出来的 java 类,它间接了继 承了 HttpServlet 类。也就是说,翻译出来的是一个 Servlet 程序 

 

 总结:通过翻译的 java 源代码我们就可以得到结果:jsp 就是 Servlet 程序。

1.4、jsp的三种语法

1.4.1、jsp头部的page指令

jsp 的 page 指令可以修改 jsp 页面中一些重要的属性,或者行为。

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

language 属性表示 jsp 翻译后是什么语言文件。暂时只支持 java。 
contentType 属性表示 jsp 返回的数据类型是什么。也是源码中 response.setContentType()参数值 
pageEncoding 属性表示当前 jsp 页面文件本身的字符集。 
import 属性 跟 java源代码中一样。用于导包,导类。

以下两个属性是给 out 输出流使用
autoFlush 属性 设置当 out 输出流缓冲区满了之后,是否自动刷新缓冲区。默认值是 true。 
buffer 属性 设置 out 缓冲区的大小。默认是 8kb

errorPage 属性 设置当 jsp 页面运行时出错,自动跳转去的错误页面路径。

errorPage 表示错误后自动跳转去的路径
这个路径一般都是以斜杠打头,它表示请求地址为 http://ip:port/工程路径/
映射到代码的 Web 目录


isErrorPage 属性 设置当前 jsp 页面是否是错误信息页面。默认是 false。如果是 true 可以 获取异常信息。 
session 属性 设置访问当前 jsp 页面,是否会创建 HttpSession 对象。默认是 true。
extends 属性 设置 jsp 翻译出来的 java 类默认继承谁。

1.4.2、jsp中的常用脚本

1.4.2.1、声明脚本(极少使用)

声明脚本的格式是

<%! 声明 java 代码 %>

作用:可以给 jsp 翻译出来的 java 类定义属性和方法甚至是静态代码块。内部类等。

1.4.2.2、表达式脚本

表达式脚本的格式是:<%=表达式%>
表达式脚本的作用是:的 jsp 页面上输出数据。 

表达式脚本的特点: 
1、所有的表达式脚本都会被翻译到_jspService() 方法中 
2、表达式脚本都会被翻译成为 out.print()输出到页面上 
3、由于表达式脚本翻译的内容都在_jspService() 方法中,所以_jspService()方法中的对象都可以直接使用。 
4、表达式脚本中的表达式不能以分号结束

1.4.2.3、代码脚本

代码脚本的格式是:

    <%
        java语句
    %>

代码脚本的作用是:可以在 jsp 页面中,编写我们自己需要的功能(写的是 java 语句)。

代码脚本的特点是:
1、代码脚本翻译之后都在_jspService 方法中 
2、代码脚本由于翻译到_jspService()方法中,所以在_jspService()方法中的现有对象都可以直接使用。 
3、还可以由多个代码脚本块组合完成一个完整的 java 语句。 
4、代码脚本还可以和表达式脚本一起组合使用,在 jsp 页面上输出数据

 1.4.3、jsp中的三种注释

1、html注释

<!-- 这是html注释 -->

html 注释会被翻译到 java 源代码中。在_jspService 方法里,以 out.writer 输出到客户端。 

2、java注释

    <%--
        //这是单行java注释
        /* 这是多行java注释 */
    --%>

java 注释会被翻译到 java 源代码中。

3、jsp注释

<%-- 这是jsp注释 --%>

jsp 注释可以注掉jsp 页面中所有代码

1.5、jsp九大内置对象

jsp 中的内置对象,是指 Tomcat 在翻译 jsp 页面成为 Servlet 源代码后,内部提供的九大对象,叫内置对象。

 exception需要在声明脚本中添加 isErrorPage="true"

<%@ page contentType="text/html;charset=UTF-8" isErrorPage="true" language="java" %>

1.6、jsp四大域对象

四个域对象分别是:
pageContext (PageContextImpl 类)     当前 jsp 页面范围内有效 
request (HttpServletRequest 类)          一次请求内有效 
session (HttpSession 类)                     一个会话范围内有效(打开浏览器访问服务器,直到关闭浏览器) 
application (ServletContext 类)            整个 web 工程范围内都有效(只要 web 工程不停止,数据都在)

域对象是可以像 Map 一样存取数据的对象。四个域对象功能一样。不同的是它们对数据的存取范围。

虽然四个域对象都可以存取数据。在使用上它们是有优先顺序的。 
四个域在使用的时候,优先顺序分别是,他们从小到大的范围的顺序。 
        pageContext ====>>> request ====>>> session ====>>> application

1.7、jsp中的out 输出和 response.getWriter 输出的区别

response 中表示响应,我们经常用于设置返回给客户端的内容(输出) 
out 也是给用户做输出使用的。

由于 jsp 翻译之后,底层源代码都是使用 out 来进行输出,所以一般情况下。我们在 jsp 页面中统一使用 out 来进行输出。避 免打乱页面输出内容的顺序。

out.write() 输出字符串没有问题 
out.print() 输出任意数据都没有问题(都转换成为字符串后调用的 write 输出)

<%--
  Created by IntelliJ IDEA.
  User: zhang
  Date: 2021/12/23
  Time: 17:11
  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>
    11111<br/>
    <%
        out.write("out输出1 <br/>");
        out.write("out输出2 <br/>");

        response.getWriter().write("response输出1 <br/>");
        response.getWriter().write("response输出2 <br/>");
    %>
</body>
</html>

结果为:

分析:

 1.8、jsp的常用标签

1.8.1、jsp静态包含

<%--
  Created by IntelliJ IDEA.
  User: zhang
  Date: 2021/12/23
  Time: 17:33
  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>
    头部信息 <br>
    主体内容 <br>
    <%--
        <%@include file=""%> 就是静态包含
            file指定要包含的jsp页面的路径
            地址中第一个斜杠 / 表示为http://ip:port/工程路径/   映射到代码的web目录
        静态包含的特点:
            1.静态包含不会翻译被包含的jsp页面
            2.静态包含其实是把被包含的jsp页面的代码拷贝到包含的位置执行输出
    --%>
    <%@include file="/include/footer.jsp"%>
</body>
</html>

1.8.2、jsp动态包含

main.jsp:

<%--
  Created by IntelliJ IDEA.
  User: zhang
  Date: 2021/12/23
  Time: 17:33
  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>
    头部信息 <br>
    主体内容 <br>
    <%--
        <%@include file=""%> 就是静态包含
            file指定要包含的jsp页面的路径
            地址中第一个斜杠 / 表示为http://ip:port/工程路径/   映射到代码的web目录
        静态包含的特点:
            1.静态包含不会翻译被包含的jsp页面
            2.静态包含其实是把被包含的jsp页面的代码拷贝到包含的位置执行输出
    --%>
    <%--<%@include file="/include/footer.jsp"%>--%>

    <%--
        <jsp:include page=""></jsp:include>就是动态包含
            page指定要包含的jsp页面的路径
        动态包含的特点:
            1.静态包含会把包含的jsp页面也翻译成为java代码
            2.动态代码底层使用如下代码去调用被包含的jsp页面执行输出
                org.apache.jasper.runtime.JspRuntimeLibrary.include(request, response, "/include/footer.jsp", out, false);
            3.动态包含还可以传递参数
    --%>
    <jsp:include page="/include/footer.jsp">
        <jsp:param name="username" value="bbj"/>
        <jsp:param name="password" value="root"/>
    </jsp:include>

</body>
</html>

footer.jsp:

<%--
  Created by IntelliJ IDEA.
  User: zhang
  Date: 2021/12/23
  Time: 17:34
  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>
    页脚信息(修改) <br>
    <%= request.getParameter("username") %>
</body>
</html>

1.8.3、jsp 标签-转发

    <%--
        <jsp:forward page=""></jsp:forward> 是请求转发标签
            page属性设置请求转发的路径
    --%>
    <jsp:forward page=""></jsp:forward>

 1.9、ServletContextListener 监听器

1、Listener 监听器它是 JavaWeb 的三大组件之一。JavaWeb 的三大组件分别是:Servlet 程序、Filter 过滤器、Listener 监听器。
2、Listener 它是 JavaEE 的规范,就是接口 
3、监听器的作用是,监听某种事物的变化。然后通过回调函数,反馈给客户(程序)去做一些相应的处理。

ServletContextListener 它可以监听 ServletContext 对象的创建和销毁。 
ServletContext 对象在 web 工程启动的时候创建,在 web 工程停止的时候销毁。 
监听到创建和销毁之后都会分别调用 ServletContextListener 监听器的方法反馈。

package jakarta.servlet;

import java.util.EventListener;

/**
 * Implementations of this interface receive notifications about changes to the
 * servlet context of the web application they are part of. To receive
 * notification events, the implementation class must be configured in the
 * deployment descriptor for the web application.
 *
 * @see ServletContextEvent
 *
 * @since Servlet 2.3
 */
public interface ServletContextListener extends EventListener {

    /**
     ** Notification that the web application initialization process is starting.
     * All ServletContextListeners are notified of context initialization before
     * any filter or servlet in the web application is initialized.
     * The default implementation is a NO-OP.
     * @param sce Information about the ServletContext that was initialized
     */
    public default void contextInitialized(ServletContextEvent sce) {
    }

    /**
     ** Notification that the servlet context is about to be shut down. All
     * servlets and filters have been destroyed before any
     * ServletContextListeners are notified of context destruction.
     * The default implementation is a NO-OP.
     * @param sce Information about the ServletContext that was destroyed
     */
    public default void contextDestroyed(ServletContextEvent sce) {
    }
}

contextInitialized()方法在ServletContext对象创建之后马上调用,做初始化

contextDestroyed()方法在ServletContext对象销毁之后调用

如何使用 ServletContextListener 监听器监听 ServletContext 对象。 
使用步骤如下: 
        1、编写一个类去实现 ServletContextListener 
        2、实现其两个回调方法 
        3、到 web.xml 中去配置监听器

监听器实现类:

package listenner;

import jakarta.servlet.ServletContextEvent;
import jakarta.servlet.ServletContextListener;


/**
 * @Author zhang
 * @Date 2021/12/24 19:06
 * @Version 1.0
 */
public class MyServletContextListennerImpl implements ServletContextListener {
    @Override
    public void contextInitialized(ServletContextEvent sce) {
        System.out.println("ServletContext对象被创建了");
    }

    @Override
    public void contextDestroyed(ServletContextEvent sce) {
        System.out.println("ServletContext对象被销毁了");
    }
}

web.xml中的配置:

 <!--配置监听器-->
    <listener>
        <listener-class>listenner.MyServletContextListennerImpl</listener-class>
    </listener>

2、EL表达式

2.1、什么是 EL 表达式,EL 表达式的作用

EL 表达式的全称是:Expression Language。是表达式语言。 
EL 表达式的什么作用:EL 表达式主要是代替 jsp 页面中的表达式脚本在 jsp 页面中进行数据的输出。 
因为 EL 表达式在输出数据的时候,要比 jsp 的表达式脚本要简洁很多。

EL 表达式的格式是:${表达式} 
EL 表达式在输出 null 值的时候,输出的是空串。
jsp 表达式脚本输出 null 值的时候,输出的是 null 字符串。

<body>
    <%
        request.setAttribute("key","value");
    %>
    表达式脚本输出key的值是:<%= request.getAttribute("key") %> <br/>
    EL表达式输出key的值是:${key}    <br/>
    表达式脚本输出key的值是:<%= request.getAttribute("key1") %> <br/>  //输出null
    EL表达式输出key的值是:${key1}  //输出为空
</body>

2.2、EL 表达式搜索域数据的顺序

EL 表达式主要是在 jsp 页面中输出数据。 
主要是输出域对象中的数据。

当四个域中都有相同的 key 的数据的时候,EL 表达式会按照四个域的从小到大的顺序去进行搜索,找到就输出。

从小到大的顺序:pageContext    request    session    application(session在浏览器关闭后数据才会消失)

2.3、EL表达式输出 Bean 的普通属性,数组属性。List 集 合属性,map 集合属性

需求——输出 Person 类中普通属性,数组属性。list 集合属性和 map 集合属性。

Person类

package pojo;

import java.util.Arrays;
import java.util.List;
import java.util.Map;

/**
 * @Author zhang
 * @Date 2021/12/24 19:51
 * @Version 1.0
 */
public class Person {
    //需求:输出 Person 类中普通属性,数组属性。list 集合属性和 map 集合属性。
    private String name;
    private String[] phone;
    private List<String> cities;
    private Map<String, Object> map;
    private int age;

    public int getAge() {
        return age;
    }

    public Person() {
    }

    public Person(String name, String[] phone, List<String> cities, Map<String, Object> map) {
        this.name = name;
        this.phone = phone;
        this.cities = cities;
        this.map = map;
    }

    public String getName() {
        return name;
    }

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

    public String[] getPhone() {
        return phone;
    }

    public void setPhone(String[] phone) {
        this.phone = phone;
    }

    public List<String> getCities() {
        return cities;
    }

    public void setCities(List<String> cities) {
        this.cities = cities;
    }

    public Map<String, Object> getMap() {
        return map;
    }

    public void setMap(Map<String, Object> map) {
        this.map = map;
    }

    @Override
    public String toString() {
        return "Person{" +
                "name=" + name +
                ", phone=" + Arrays.toString(phone) +
                ", cities=" + cities +
                ", map=" + map +
                '}';
    }
}

输出界面:


<%@ page import="java.util.List" %>
<%@ page import="java.util.ArrayList" %>
<%@ page import="java.util.Map" %>
<%@ page import="java.util.HashMap" %>
<%@ page import="pojo.Person" %><%--
  Created by IntelliJ IDEA.
  User: zhang
  Date: 2021/12/24
  Time: 20:02
  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>
    <%
        Person person = new Person();
        person.setName("帅哥");
        person.setPhone(new String[]{"00001","00002","00003"});
        List<String> cities = new ArrayList<>();
        cities.add("北京");
        cities.add("伤害");
        cities.add("厦门");
        person.setCities(cities);
        Map<String,Object> map = new HashMap<>();
        map.put("key1","value1");
        map.put("key2","value2");
        map.put("key3","value3");
        person.setMap(map);
        pageContext.setAttribute("p",person);
    %>
    输出person:${p}   <br/>
    输出person的name属性:${p.name}   <br/>
    输出person的phone数组属性值:${p.phone}   <br/> <%--[Ljava.lang.String;@56389c01--%>
    输出person的phone数组某个元素属性:${p.phone[1]}   <br/> <%--00002--%>
    输出person的List属性:${p.cities}   <br/>
    输出person的List某个元素值:${p.cities[2]}   <br/>
    输出person的map属性:${p.map}   <br/>
    输出person的map中某个key的值:${p.map.key1}   <br/>
    输出person的age属性:${p.name}   <br/>
    <%--
        若在该类中没有对应属性的get方法,则无法找到该元素,报错
        若把Person类的age删掉而保留getAge方法,但return 18,则浏览器界面输出也为18
    --%>
</body>
</html>

2.4、EL 表达式——运算

语法:${ 运算表达式 } 

2.4.1、关系运算

2.4.2、逻辑运算 

 2.4.3、算数运算

 2.4.4、empty运算

empty 运算可以判断一个数据是否为空,如果为空,则输出 true,不为空输出 false。 
以下几种情况为空: 
        1、值为 null 值的时候,为空 
        2、值为空串的时候,为空 
        3、值是 Object 类型数组,长度为零的时候 
        4、list 集合,元素个数为零 
        5、map 集合,元素个数为零

2.4.5、三元运算

表达式 1?表达式 2:表达式 3 
如果表达式 1 的值为真,返回表达式 2 的值,如果表达式 1 的值为假,返回表达式 3 的值。

2.4.6、" . "运算和 [ ] 中括号运算符

.点运算,可以输出 Bean 对象中某个属性的值。 
[]中括号运算,可以输出有序集合中某个元素的值。 并且[]中括号运算,还可以输出 map 集合中 key 里含有特殊字符的 key 的值。

<%
        Map<String,Object> map = new HashMap<>();
        map.put("a.a.a","aaaValue");
        map.put("b+b+b","bbbValue");
        map.put("c-c-c","cccValue");
        request.setAttribute("map",map);
    %>
    <%--中括号里面可用用单引号,也可以用双引号--%>
    ${ map['a.a.a'] } <br/>
    ${ map['b+b+b'] } <br/>
    ${ map["c-c-c"] } <br/>

2.4.7、EL 表达式的 11 个隐含对象

EL 个达式中 11 个隐含对象,是 EL 表达式中自己定义的,可以直接使用。

        变量                                                  类型                                     作用 
pageContext                         PageContextImpl                              它可以获取 jsp 中的九大内置对象 


pageScope                           Map<String,Object>                          它可以获取 pageContext 域中的数据 
requestScope                       Map<String,Object>                          它可以获取 Request 域中的数据 
sessionScope                      Map<String,Object>                           它可以获取 Session 域中的数据 
applicationScope                 Map<String,Object>                           它可以获取 ServletContext 域中的数据 


param                                  Map<String,String>                             它可以获取请求参数的值 
paramValues                       Map<String,String[]>                           它也可以获取请求参数的值,获取多个值的时候使用。 


header                                Map<String,String>                             它可以获取请求头的信息 
headerValues                     Map<String,String[]>                           它可以获取请求头的信息,它可以获取多个值的情况 


cookie                                 Map<String,Cookie>                           它可以获取当前请求的 Cookie 信息
initParam                            Map<String,String>                              它可以获取在 web.xml 中配置的<context-param>上下文参数

2.4.7.1、EL 获取四个特定域中的属性

pageScope          ======  pageContext 域 
requestScope      ======  Request 域 
sessionScope      ======  Session 域 
applicationScope ======  ServletContext 域

<%
        pageContext.setAttribute("key1","pageContext1");
        pageContext.setAttribute("key2","pageContext2");
        request.setAttribute("key2","request2");
        session.setAttribute("key2","session2");
        application.setAttribute("key2","application2");
    %>
    ${ key2 } <%--只能从小到大范围的域中选取含key2的最小域--%>
    ${ requestScope.key2 } <%--可以指定域--%>

2.4.7.2、pageContext 对象的使用

1. 协议: 
2. 服务器 ip: 
3. 服务器端口: 
4. 获取工程路径: 
5. 获取请求方法: 
6. 获取客户端 ip 地址: 
7. 获取会话的 id 编号:

    <%--
        request.getScheme() 获取请求的协议
        request.getServerName() 获取请求的服务器ip或域名
        request.getServerPort() 获取请求的端口号
        request.getContextPath() 获取当前工程路径
        request.getMethod() 获取请求的方式(GET 或 POST)
        request.getRemoteHost() 获取客户端的ip地址
        session.getId() 获取会话的唯一标识id
    --%>
    <%=request.getScheme()%> <br/>  <%--http--%>
    <%=request.getServerName()%>
    <%=request.getServerPort()%>
    <%=request.getContextPath()%>
    <%=request.getMethod()%>
    <%=request.getRemoteHost()%>
    <%=session.getId()%>
    <hr/>
    1. 协议:${pageContext.request.scheme} <br/>
    2. 服务器 ip:${pageContext.request.serverName} <br/>
    3. 服务器端口:${pageContext.request.serverPort} <br/>
    4. 获取工程路径:${pageContext.request.contextPath} <br/>
    5. 获取请求方法:${pageContext.request.method} <br/>
    6. 获取客户端 ip 地址:${pageContext.request.remoteHost} <br/>
    7. 获取会话的 id 编号:${pageContext.session.id} <br/>

2.4.7.3、EL 表达式其他隐含对象的使用

param                                  Map<String,String>                             它可以获取请求参数的值 
paramValues                       Map<String,String[]>                           它也可以获取请求参数的值,获取多个值的时候使用。 

示例代码:

<%--
        浏览器地址为http://localhost:8080/11_EL_JSTL/other_el_obj.jsp?username=zyj&password=10086
    --%>
    输出请求参数username的值:${ param.username } <br/>
    输出请求参数password>的值:${ param.password } <br/>
    <hr/>
    <%--浏览器地址为http://localhost:8080/11_EL_JSTL/other_el_obj.jsp?&hobby=java&hobby=cpp--%>
    输出请求参数hobby的第0个位置的值:${ paramValues.hobby[0] } <br/>
    输出请求参数hobby的第1个位置的值:${ paramValues.hobby[1] } <br/>

header                                Map<String,String>                             它可以获取请求头的信息 
headerValues                     Map<String,String[]>                           它可以获取请求头的信息,它可以获取多个值的情况 

 输出请求头【User-Agent】的值:${ header['User-Agent'] } <br/>
    输出请求头【Connection】的值:${ header['Connection'] } <br/>
    输出请求头【User-Agent】的值:${ headerValues['User-Agent'][0] } <br/>

cookie                                 Map<String,Cookie>                           它可以获取当前请求的 Cookie 信息

获取Cookie的名称:${ cookie.JSESSIONID.name } <br/>
    获取Cookie的值:${ cookie.JSESSIONID.value } <br/>

initParam                            Map<String,String>                              它可以获取在 web.xml 中配置的<context-param>上下文参数

3、JSTL标签库

JSTL 标签库 全称是指 JSP Standard Tag Library JSP 标准标签库。是一个不断完善的开放源代码的 JSP 标 签库。
EL 表达式主要是为了替换 jsp 中的表达式脚本,而标签库则是为了替换代码脚本。这样使得整个 jsp 页面 变得更佳简洁。

JSTL 由五个不同功能的标签库组成:

CORE 标签库 
<%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core" %> 
XML 标签库
<%@ taglib prefix="x" uri="http://java.sun.com/jsp/jstl/xml" %> 
FMT 标签库 
<%@ taglib prefix="fmt" uri="http://java.sun.com/jsp/jstl/fmt" %> 
SQL 标签库 
<%@ taglib prefix="sql" uri="http://java.sun.com/jsp/jstl/sql" %> 
FUNCTIONS 标签库 
<%@ taglib prefix="fn" uri="http://java.sun.com/jsp/jstl/functions" %>

3.1、JSTL 标签库的使用步骤

1、先导入 jstl 标签库的 jar 包。 taglibs-standard-impl-1.2.1.jar taglibs-standard-spec-1.2.1.jar 
2、第二步,使用 taglib 指令引入标签库。

3.2、core 核心库使用

3.2.0、使用过程遇到的问题

无法解析路径:无法在web.xml或使用此应用程序部署的jar文件中解析绝对uri:[http://java.sun.com/jsp/jstl/core] - Vlats - 博客园

导完之后还是出错:org.apache.jasper.JasperException: org.apache.jasper.JasperException: java.lang.ClassNotFoundExcepti_贝西晨的博客-CSDN博客_org.apache.jasper.jasperexception

最终解决办法:导入的包不兼容:jstl - java.lang.NoClassDefFoundError: javax/servlet/jsp/tagext/TagLibraryValidator - Stack Overflow 

3.2.1、<c:set/>(使用很少)

作用:set 标签可以往域中保存数据

<%--
        <c:set/>(使用很少)
            作用:set 标签可以往域中保存数据
            域对象.setAttribute(key,value);
            scope属性设置保存到哪个域
                page表示PageContext域(默认值)
                request表示Request域
                session表示Session域
                application表示ServletContext域
            var属性设置key
            value属性设置值
    --%>
    保存之前:${ requestScope.abc } <br>
    <c:set scope="request" var="abc" value="abcValue"/>
    保存之后:${ requestScope.abc } <br>

3.2.2、<c:if />

if 标签用来做 if 判断。

<%--
        if 标签用来做 if 判断。
            test属性表示判断的条件
    --%>
    <c:if test="${12 == 12}">
        <h1>12等于12</h1>
    </c:if>

3.2.3、<c:choose> <c:when> <c:otherwise>标签

作用:多路判断。跟 switch ... case .... default 非常接近

<%--
iii.<c:choose> <c:when> <c:otherwise>标签
作用:多路判断。跟switch ... case .... default非常接近

choose标签开始选择判断
when标签表示每一种判断情况
    test属性表示当前这种判断情况的值
otherwise标签表示剩下的情况

<c:choose> <c:when> <c:otherwise>标签使用时需要注意的点:
    1、标签里不能使用html注释,要使用jsp注释
    2、when标签的父标签一定要是choose标签
--%>
<%
    request.setAttribute("height", 180);
%>
<c:choose>
    <%-- 这是html注释 --%>
    <c:when test="${ requestScope.height > 190 }">
        <h2>小巨人</h2>
    </c:when>
    <c:when test="${ requestScope.height > 180 }">
        <h2>很高</h2>
    </c:when>
    <c:when test="${ requestScope.height > 170 }">
        <h2>还可以</h2>
    </c:when>
    <c:otherwise>
        <c:choose>
            <c:when test="${requestScope.height > 160}">
                <h3>大于160</h3>
            </c:when>
            <c:when test="${requestScope.height > 150}">
                <h3>大于150</h3>
            </c:when>
            <c:when test="${requestScope.height > 140}">
                <h3>大于140</h3>
            </c:when>
            <c:otherwise>
                其他小于140
            </c:otherwise>
        </c:choose>
    </c:otherwise>
</c:choose>

3.2.4、<c:forEach />

1、遍历1到10并输

2、遍历Object数组

3、遍历Map集合

4、遍历 List 集合---list 中存放 Student 类,有属性:编号,用户名,密码,年龄, 电话信息

<%@ page import="java.util.HashMap" %>
<%@ page import="java.util.Map" %>
<%@ page import="pojo.Student" %>
<%@ page import="java.util.ArrayList" %>
<%@ page import="java.util.List" %>

<%--
  Created by IntelliJ IDEA.
  User: zhang
  Date: 2021/12/25
  Time: 11:45
  To change this template use File | Settings | File Templates.
--%>
<%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core" %>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<%@page isELIgnored="false"%>
<html>
<head>
    <title>Title</title>
    <style type="text/css">
        table{
            width: 500px;
            border: 1px solid red;
            border-collapse: collapse;
        }
        tr,td{
            border: 1px solid red;
        }
    </style>
</head>
<body>
    <%--
        遍历1到10的情况,输出
            begin设置开始的索引
            end设置结束的索引
            var属性设置循环的变量
    --%>
    <table border="1">
        <c:forEach begin="1" end="10" var="i">
            <tr>
                <td>第${i}行</td>
            </tr>
        </c:forEach>
    </table>
    <hr/>
    <%--
        遍历Object数组
            items表示遍历的数据源(遍历的集合)
            var 表示当前遍历到的数据
    --%>
    <%
        request.setAttribute("arr",new Object[]{"00000","00001","00002"});
    %>
    <c:forEach items="${requestScope.arr}" var="item">
        ${item} <br/>
    </c:forEach>
    <hr/>
    <%--

    --%>
    <%
        Map<String,Object> map = new HashMap<>();
        map.put("key1","value1");
        map.put("key2","value2");
        map.put("key3","value3");
        //for(Map.Entry<String, Object> entry : map.entrySet()){}
        request.setAttribute("map",map);
    %>
    <c:forEach items="${requestScope.map}" var="entry">
        ${entry} <br>
        ${entry.key} = ${entry.value} <br>
    </c:forEach>
    <hr/>

    <%--
        遍历 List 集合---list 中存放 Student 类,有属性:编号,用户名,密码,年龄, 电话信息
    --%>
    <%
        List<Student> studentList = new ArrayList<>();
        for (int i = 0; i < 10; i++) {
            studentList.add(new Student(i,"username"+i,"pass"+i,18+i,"phone"+i));
        }
        request.setAttribute("stus",studentList );
    %>
    <table>
        <tr>
            <td>编号</td>
            <td>用户名</td>
            <td>密码</td>
            <td>年龄</td>
            <td>电话</td>
            <td>操作</td>
        </tr>
        <%--
            item 表示遍历的集合
            var 表示遍历到的数据
            begin 表示遍历开始的索引
            end 表示结束的索引
            step 表示遍历的不长
            varstatus 表示当前遍历到的数据的状态
        --%>
        <c:forEach begin="2" end="8" step="2" items="${requestScope.stus}" var="stu">
            <tr>
                <td>${stu.id}</td>
                <td>${stu.username}</td>
                <td>${stu.password}</td>
                <td>${stu.age}</td>
                <td>${stu.phone}</td>
                <td>删除、修改</td>
            </tr>
        </c:forEach>

    </table>
</body>
</html>

4、文件的上传和下载

4.0、使用过程遇到的坑

tomcat10的文件头是Jakarta,导致无法解析ServletFileUpload.isMultipartContent(req)。解决办法:换低版本tomcat

4.1、文件的上传介绍

1、要有一个 form 标签,method=post 请求 
2、form 标签的 encType 属性值必须为 multipart/form-data 值 
3、在 form 标签中使用 input type=file 添加上传的文件 
4、编写服务器代码(这里使用Servlet 程序)接收,处理上传的数据。 

encType=multipart/form-data 表示提交的数据,以多段(每一个表单项一个数据段)的形式进行拼 接,然后以二进制流的形式发送给服务器

4.1.1、文件上传、HTTP协议说明

 4.1.2、commons-fileupload.jar 常用 API 介绍说明

commons-fileupload.jar 需要依赖 commons-io.jar 这个包,所以两个包我们都要引入。

常用:

ServletFileUpload 类,用于解析上传的数据。 FileItem 类,表示每一个表单项。 
boolean ServletFileUpload.isMultipartContent(HttpServletRequest request); 判断当前上传的数据格式是否是多段的格式。 
public List<FileItem> parseRequest(HttpServletRequest request) 解析上传的数据 
boolean FileItem.isFormField() 判断当前这个表单项,是否是普通的表单项。还是上传的文件类型。 true 表示普通类型的表单项 false 表示上传的文件类型 
String FileItem.getFieldName() 获取表单项的 name 属性值
String FileItem.getString() 获取当前表单项的值。 
String FileItem.getName(); 获取上传的文件名 
void FileItem.write( file ); 将上传的文件写到 参数 file 所指向抽硬盘位置 。

4.1.3、文件上传的实现

表单upload.jsp:

<%--
  Created by IntelliJ IDEA.
  User: zhang
  Date: 2021/12/25
  Time: 20:25
  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>
    <form action="http://localhost:8080/12_EL_JSTL/uploadServlet" method="post" enctype="multipart/form-data">
        用户名:<input type="text" name="username"/> <br>
        头像:<input type="file" name="photo"> <br>
        <input type="submit" value="上传">
    </form>
</body>
</html>

实现代码UploadServlet.java:

package servlet;

import org.apache.commons.fileupload.FileItem;
import org.apache.commons.fileupload.FileItemFactory;
import org.apache.commons.fileupload.FileUploadException;
import org.apache.commons.fileupload.disk.DiskFileItemFactory;
import org.apache.commons.fileupload.servlet.ServletFileUpload;

import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.File;
import java.io.IOException;
import java.util.List;

/**
 * @Author zhang
 * @Date 2021/12/25 20:30
 * @Version 1.0
 */
public class UploadServlet extends HttpServlet {
    /**
     * 用来处理上传的数据
     *
     * @param req
     * @param resp
     * @throws ServletException
     * @throws IOException
     */
    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        //System.out.println("文件已经上传过来");
        //1 判断上传的数据是否为多段数据(只有是多段数据,才是文件上传的)
        if (ServletFileUpload.isMultipartContent(req)) {
            //创建FileItemFactory工厂实现类
            FileItemFactory fileItemFactory = new DiskFileItemFactory();
            //创建用于解析上传数据的工具类ServletFileUpload类
            ServletFileUpload servletFileUpload = new ServletFileUpload(fileItemFactory);

            try {
                //解析上传的数据,得到每一个表单项FileItem
                List<FileItem> list = servletFileUpload.parseRequest(req);
                //循环判断meii=yige表单项是普通类型还是上传的文件
                for (FileItem fileItem : list) {
                    if (fileItem.isFormField()) {
                        //普通表单项
                        System.out.println("表单项的name属性值:" + fileItem.getFieldName());
                        //参数UTF-8避免乱码
                        System.out.println("表单项的value属性值:" + fileItem.getString("UTF-8"));
                    } else {
                        //上传的文件
                        System.out.println("表单项的name属性值:" + fileItem.getFieldName());
                        System.out.println("上传的文件名:" + fileItem.getName());

                        fileItem.write(new File("D:\\Java\\Javaweb\\myproject\\mypicture\\a.jpg" + fileItem.getName()));

                    }
                }

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

4.2、文件上传

4.2.1、实现(中文乱码)

package servlet;

import org.apache.commons.io.IOUtils;

import javax.servlet.ServletContext;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;

/**
 * @Author zhang
 * @Date 2021/12/26 19:23
 * @Version 1.0
 */
public class Download extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        //获取要下载的文件名
        String downloadFileName = "a.jpg";

        //读取要下载的文件内容(通过ServletContext对象读取)
        ServletContext servletContext = getServletContext();

        //获取要下载的文件类型
        String mimeType = servletContext.getMimeType("/file/" + downloadFileName);
        //System.out.println("要下载的文件类型为:" + mimeType);

        //在回传前,通过响应头告诉客户端返回的数据类型
        resp.setContentType(mimeType);

        //告诉客户端收到的数据是用于下载使用(使用响应头)
        //Content-Disposition响应头,表示收到的数据怎么处理
        //attachment表示附件,表示下载使用
        //filename= 表示指定下载的文件名,默认不支持中文
        //resp.setHeader("Content-Disposition","attachment;filename=mypicture.jpg");
        resp.setHeader("Content-Disposition","attachment;filename=" + downloadFileName);

        //获取输入流
        InputStream resourceAsStream = servletContext.getResourceAsStream("/file/" + downloadFileName);

        //获取响应的输出流
        OutputStream outputStream = resp.getOutputStream();

        //读取输入流全部数据,输出给输出流
        IOUtils.copy(resourceAsStream,outputStream);


    }
}

4.2.2、中文乱码解决----URLEncoder编码(IE和谷歌浏览器)

修改设置响应头

package servlet;

import org.apache.commons.io.IOUtils;

import javax.servlet.ServletContext;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.URLEncoder;

/**
 * @Author zhang
 * @Date 2021/12/26 19:23
 * @Version 1.0
 */
public class Download extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        //获取要下载的文件名
        String downloadFileName = "a.jpg";

        //读取要下载的文件内容(通过ServletContext对象读取)
        ServletContext servletContext = getServletContext();

        //获取要下载的文件类型
        String mimeType = servletContext.getMimeType("/file/" + downloadFileName);
        //System.out.println("要下载的文件类型为:" + mimeType);

        //在回传前,通过响应头告诉客户端返回的数据类型
        resp.setContentType(mimeType);

        //告诉客户端收到的数据是用于下载使用(使用响应头)
        //Content-Disposition响应头,表示收到的数据怎么处理
        //attachment表示附件,表示下载使用
        //filename= 表示指定下载的文件名,默认不支持中文
        //resp.setHeader("Content-Disposition","attachment;filename=mypicture.jpg");
        //url编码是把汉字转换为%xx%xx的格式
        resp.setHeader("Content-Disposition","attachment;filename=" + URLEncoder.encode("我的图片.jpg","UTF-8"));

        //获取输入流
        InputStream resourceAsStream = servletContext.getResourceAsStream("/file/" + downloadFileName);

        //获取响应的输出流
        OutputStream outputStream = resp.getOutputStream();

        //读取输入流全部数据,输出给输出流
        IOUtils.copy(resourceAsStream,outputStream);


    }
}

4.2.3、中文乱码解决------BASE64编码(火狐)

首先了解以下Base64编码的编码和解码操作

package base64;

import java.io.UnsupportedEncodingException;
import java.util.Base64;

/**
 * @Author zhang
 * @Date 2021/12/26 20:09
 * @Version 1.0
 */
public class base64Test {

    public static void main(String[] args) throws UnsupportedEncodingException {
        String content = "这是需要Base64编码的内容";
        //创建一个Base64编码器
        Base64.Encoder encoder = Base64.getEncoder();
        //编码
        String encodeToString = encoder.encodeToString(content.getBytes("UTF-8"));
        System.out.println(encodeToString); //6L+Z5piv6ZyA6KaBQmFzZTY057yW56CB55qE5YaF5a65
        //创建解码器
        Base64.Decoder decoder = Base64.getDecoder();
        //解码
        byte[] bytes = decoder.decode(encodeToString);
        System.out.println(new String(bytes,"UTF-8")); //这是需要Base64编码的内容
    }
}

解决中文乱码问题:

需要把请求头 Content-Disposition: attachment; filename=中文名 
编码成为:Content-Disposition: attachment; filename==?charset?B?xxxxx?= =?charset?B?xxxxx?= 
=?                表示编码内容的开始
charset        表示字符集
B                 表示BASE64编码
xxxx            表示文件名BASE64编码后的内容
?=               表示编码内容的结束

//中文乱码解决------BASE64编码(火狐)
        resp.setHeader("Content-Disposition","attachment;filename=?UTF-8?B?" + Base64.getEncoder().encodeToString("我的图片.jpg".getBytes("UTF-8")) + "?=");

4.2.4、最终版

if(req.getHeader("User-Agent").contains("Firefox")){
            //如果是火狐浏览器使用base64编码
            resp.setHeader("Content-Disposition","attachment;filename=?UTF-8?B?" + Base64.getEncoder().encodeToString("我的图片.jpg".getBytes("UTF-8")) + "?=");
        }else{
            //不是火狐,使用URL编码
            resp.setHeader("Content-Disposition","attachment;filename=mypicture.jpg");
        }

5、MVC概念

MVC 全称:Model 模型、 View 视图、 Controller 控制器。 
MVC 最早出现在 JavaEE 三层中的 Web 层,它可以有效的指导 Web 层的代码如何有效分离,单独工作。
View 视图:只负责数据和界面的显示,不接受任何与显示数据无关的代码,便于程序员和美工的分工合作—— JSP/HTML。
Controller 控制器:只负责接收请求,调用业务层的代码处理请求,然后派发页面,是一个“调度者”的角色——Servlet。 转到某个页面。或者是重定向到某个页面。 
Model 模型:将与业务逻辑相关的数据封装为具体的 JavaBean 类,其中不掺杂任何与数据处理相关的代码—— JavaBean/domain/entity/pojo。 

MVC 是一种思想 
MVC 的理念是将软件代码拆分成为组件,单独开发,组合使用(目的还是为了降低耦合度)。

MVC 的作用是为了降低耦合。让代码合理分层。方便后期升级和维护。

 6、cookie

6.1、什么是Cookie

1、Cookie 翻译过来是饼干的意思。 
2、Cookie 是服务器通知客户端保存键值对的一种技术。 
3、客户端有了 Cookie 后,每次请求都发送给服务器。 
4、每个 Cookie 的大小不能超过 4kb

注意:Cookie不支持中文,对于 Version 0 cookie,值不应包含空格、方括号、圆括号、等号、逗号、双引号、斜杠、问号、at 符号、冒号和分号。空值在所有浏览器上的行为不一定相同。 如果使用二进制值,则可能需要使用 BASE64 编码。 

6.2、如何创建Cookie

public class CookieServlet extends BaseServlet{

    protected void createCookie(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        //1.创建Cookie对象
        Cookie cookie = new Cookie("key1","value1");
        //2.通知客户端保存Cookie
        resp.addCookie(cookie);

        //1.创建Cookie对象
        Cookie cookie1 = new Cookie("key3","value3");
        //2.通知客户端保存Cookie
        resp.addCookie(cookie1);
        
        resp.getWriter().write("Cookie创建成功");

    }
}

6.3、服务器如何获取Cookie

服务器获取客户端的 Cookie 只需要一行代码:req.getCookies():Cookie[]

 cookie 的工具类:

package utils;

import javax.servlet.http.Cookie;

/**
 * @Author zhang
 * @Date 2021/12/27 22:17
 * @Version 1.0
 */
public class CookieUtils {


    /**
     * 查找指定名称的Cookie对象
     * @param name
     * @param cookies
     * @return
     */
    public static Cookie findCookie(String name,Cookie[] cookies){
        if(name == null || cookies == null || cookies.length==0){
            return null;
        }
        for (Cookie cookie : cookies) {
            if(name.equals(cookie.getName())){
                return cookie;
            }
        }
        return null;
    }
}

 servlet中的代码

package Servlet;

import utils.CookieUtils;

import javax.servlet.ServletException;
import javax.servlet.http.Cookie;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;

/**
 * @Author zhang
 * @Date 2021/12/27 21:43
 * @Version 1.0
 */
public class CookieServlet extends BaseServlet{

    protected void createCookie(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        //1.创建Cookie对象
        Cookie cookie = new Cookie("key1","value1");
        //2.通知客户端保存Cookie
        resp.addCookie(cookie);

        //1.创建Cookie对象
        Cookie cookie1 = new Cookie("key3","value3");
        //2.通知客户端保存Cookie
        resp.addCookie(cookie1);

        resp.getWriter().write("Cookie创建成功");
    }

    protected void getCookie(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        Cookie[] cookies = req.getCookies();

        //获取所有的Cookie
        for (Cookie cookie : cookies) {
            resp.getWriter().write(cookie.getName() + "---------" +cookie.getValue() + "<br/>");
        }

        Cookie iWantCookie = CookieUtils.findCookie("key1",cookies);
        if(iWantCookie != null){
            resp.getWriter().write("找到了需要的cookie");
        }
    }
}

6.4、Cookie值的修改

6.4.1、方案一: 

1、先创建一个要修改的同名(指的就是 key)的 Cookie 对象 
2、在构造器,同时赋于新的 Cookie 值。 
3、调用 response.addCookie( Cookie );

        //方案一:
        //1、先创建一个要修改的同名(指的就是 key)的 Cookie 对象
        //2、在构造器,同时赋于新的 Cookie 值。
        Cookie cookie = new Cookie("key1","V1");
        //3、调用 response.addCookie( Cookie );通知客户端保存修改
        resp.addCookie(cookie);
        resp.getWriter().write("key1的Cookie值已经修改好");

6.4.2、方案二

1、先查找到需要修改的 Cookie 对象 
2、调用 setValue()方法赋于新的 Cookie 值。 
3、调用 response.addCookie()通知客户端保存修改

//方案二:
        //1、先查找到需要修改的 Cookie 对象
        Cookie cookie = CookieUtils.findCookie("key3",req.getCookies());
        //2、调用 setValue()方法赋于新的 Cookie 值。
        if(cookie != null){
            cookie.setValue("V3");
        }
        //3、调用 response.addCookie()通知客户端保存修改
        resp.addCookie(cookie);

6.5、浏览器查看Cookie

6.6、Cookie生命控制 

Cookie 的生命控制指的是如何管理 Cookie 什么时候被销毁(删除) 
public void setMaxAge(int expiry) 
        正数,表示在指定的秒数后过期 
        负数,表示浏览器一关,Cookie 就会被删除(默认值是-1) 
        零,表示马上删除 Cookie

protected void defaultLife(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        Cookie cookie = new Cookie("defaultLife","defaultLife");
        cookie.setMaxAge(-1); //设置存活时间
        resp.addCookie(cookie);
    }

    protected void deleteNow(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        //先找到要删除的Cookie对象
        Cookie cookie = CookieUtils.findCookie("key1",req.getCookies());
        //调用setMaxAge(0);
        if(cookie != null){
            cookie.setMaxAge(0); //表示马上删除,不需要等待浏览器关闭
        }
        //调用response.addCookie(cookie)
        resp.addCookie(cookie);
        resp.getWriter().write("key1的Cookie已经删除");
    }

    protected void life3600(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        Cookie cookie = new Cookie("life3600", "life3600");
        cookie.setMaxAge(60 * 60);  //设置一小时后被删除/无效
        resp.addCookie(cookie);
        resp.getWriter().write("已经创建了一个存活一小时的Cookie");
    }

6.7、Cookie有效路径Path的设置

Cookie 的 path 属性可以有效的过滤哪些 Cookie 可以发送给服务器。哪些不发。

path 属性是通过请求的地址来进行有效的过滤。

CookieA path=/工程路径
CookieB path=/工程路径/abc

请求地址如下: http://ip:port/工程路径/a.html 
        CookieA 发送 
        CookieB 不发送 
http://ip:port/工程路径/abc/a.html 
        CookieA 发送 
        CookieB 发送

 protected void testPath(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        Cookie cookie = new Cookie("path1","path1");
        //req.getContextPath() 得到工程路径
        cookie.setPath(req.getContextPath() + "/abc"); //得到   /工程路径/abc
        resp.addCookie(cookie);
        resp.getWriter().write("创建了一个带有工程路径的Cookie");
    }

6.8、Cookie练习--免输入用户名登录

login.jsp页面

<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
    <title>Title</title>
</head>
<body>
    <form action="http://localhost:8080/13_cookie/loginServlet" method="get">
        用户名: <input type="text" name="username" value="${cookie.username.value}"> <br>
        密码: <input type="text" name="password" > <br>
        <input type="submit" value="登录">
    </form>
</body>
</html>

LoginServlet.java

package Servlet;

import javax.servlet.ServletException;
import javax.servlet.http.Cookie;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;

/**
 * @Author zhang
 * @Date 2021/12/27 23:12
 * @Version 1.0
 */
public class LoginServlet extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        String username = req.getParameter("username");
        String password = req.getParameter("password");

        if ("admin".equals(username) && "admin".equals(password)) {
            //登录成功
            Cookie cookie = new Cookie("username", username);
            cookie.setMaxAge(60 * 60 * 24 * 7); //当前Cookie一周有效
            resp.addCookie(cookie);
            System.out.println("登录成功");
        } else {
            //登录失败
            System.out.println("登录失败");
        }
    }
}

7、Session会话

7.1、什么是Session会话

1、Session 就是一个接口(HttpSession)。 
2、Session 就是会话。它是用来维护一个客户端和服务器之间关联的一种技术。 
3、每个客户端都有自己的一个 Session 会话。 
4、Session 会话中,我们经常用来保存用户登录之后的信息。

7.2、如何创建Session和获取(id号,是否为新)

如何创建和获取 Session。它们的 API 是一样的。

request.getSession() 
        第一次调用是:创建 Session 会话 
        之后调用都是:获取前面创建好的 Session 会话对象。

isNew(); 判断到底是不是刚创建出来的(新的) 
        true 表示刚创建 
        false 表示获取之前创建

每个会话都有一个身份证号。也就是 ID 值。而且这个 ID 是唯一的。 
getId() 得到 Session 的会话 id 值。

注意:若isNew();一直返回false,则可能是有缓存了,将Cookie清空即可

protected void createOrGetSession(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        //创建和获取Session会话对象
        HttpSession session = req.getSession();

        //判断当前的Session会话是否是新创建出来的
        //若isNew();一直返回false,则可能是有缓存了,将Cookie清空即可
        boolean isNew = session.isNew();

        //获取Session会话的唯一表示
        String id = session.getId();

        resp.getWriter().write("得到的Session,它的id是:" + id + "<br/>");
        resp.getWriter().write("得到的Session是否是新创建的:" + isNew + "<br/>");

    }

7.3、Session域的获取

/**
     * 往Session域中保存数据
     * @param req
     * @param resp
     * @throws ServletException
     * @throws IOException
     */
    protected void setAttribute(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        req.getSession().setAttribute("key1","value1");
        resp.getWriter().write("已经往Session域中保存了数据");
    }

    /**
     * 获取Session域中的数据
     * @param req
     * @param resp
     * @throws ServletException
     * @throws IOException
     */
    protected void getAttribute(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        Object attribute = req.getSession().getAttribute("key1");
        resp.getWriter().write("从Session域中取出key1的数据是" + attribute);

    }

7.4、Session生命周期控制

public void setMaxInactiveInterval(int interval) 设置 Session 的超时时间(以秒为单位),超过指定的时长,Session 就会被销毁。
        值为正数的时候,设定 Session 的超时时长。 
        负数表示永不超时(极少使用)

public int getMaxInactiveInterval()获取 Session 的超时时间

public void invalidate() 让当前 Session 会话马上超时无效。

Session 默认的超时时间长为 30 分钟。 
因为在 Tomcat 服务器的配置文件 web.xml中默认有以下的配置,它就表示配置了当前 Tomcat 服务器下所有的 Session 超时配置默认时长为:30 分钟。

<session-config> 
    <session-timeout>30</session-timeout> 
</session-config>

如果说。你希望你的 web 工程,默认的 Session 的超时时长为其他时长。你可以在你自己的 web.xml 配置文件中做 以上相同的配置。就可以修改你的 web 工程所有 Seession 的默认超时时长。

<!--表示当前web工程创建出来的所有Session默认是20分钟超时时长-->
    <session-config>
        <session-timeout>20</session-timeout>
    </session-config>

如果你想只修改个别 Session 的超时时长。就可以使用下面的 API。setMaxInactiveInterval(int interval)来进行单独的设 置。
session.setMaxInactiveInterval(int interval)单独设置超时时长。

protected void defaultLife(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        //获取Session的默认超时时长
        int maxInactiveInterval = req.getSession().getMaxInactiveInterval();

        resp.getWriter().write("Session的默认超时时长为" + maxInactiveInterval + "秒");
    }

    protected void life3(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        //获取Session对象
        HttpSession session = req.getSession();

        //设置当前Session三秒后超时
        session.setMaxInactiveInterval(3);

        resp.getWriter().write("当前Session已经设置为3秒后超时");
    }

    protected void deleteNow(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        //获取Session对象
        HttpSession session = req.getSession();

        //让Session会话马上超时
        session.invalidate();

        resp.getWriter().write("Sessuion已经设置为马上销毁");
    }

7.5、浏览器和 Session 之间关联的技术内幕

Session 技术,底层其实是基于 Cookie 技术来实现的。 

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值