IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> Java知识库 -> 解析Java Web之Servlet -> 正文阅读

[Java知识库]解析Java Web之Servlet

Servlet

Servlet先于JSP出现

1、Servlet 是 JavaEE 规范之一。规范就是接口
2、Servlet 就 JavaWeb 三大组件之一。
三大组件分别是:Servlet 程序、Filter 过滤器、Listener 监听器
3、Servlet 是使用Java语言实现的程序,运行在服务器(支持Java语言的Web服务器或者应用服务器)上的一个 java 小程序,通常通过http它可以接收客户端发送过来的请求,并响应数据给客户端

实现Servlet程序

工程目录
在这里插入图片描述

1.编写一个类去实现Servlet接口
2.实现service方法,处理请求,并响应数据
在这里插入图片描述
3.到WEB-INF/web.xml去配置Servlet程序的访问地址

WEB—INF是web的安全目录。所谓安全目录就是客户端无法访问的目录,只有服务端可以访问,当静态资源放在WEB—INF下面的时候,外部是不能访问的

<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_4_0.xsd"
         version="4.0">

         <!--Servlet标签给服务器Tomcat配置Servlet程序-->
         <servlet>
             <!--servlet-name标签给servlet程序起一个别名,一般是类名-->
             <servlet-name>HelloServlet</servlet-name>
             <!--servlet-class是servlet程序的全类名-->
             <servlet-class>com.Servlet.HelloServlet</servlet-class>
         </servlet>
        <!--servlet-maping给selvlet标签配置访问地址-->
         <servlet-mapping>
             <!--servlet-name标签作用是告诉服务器,我当前配置的地址给哪个Servlet程序使用-->
             <servlet-name>HelloServlet</servlet-name>
             <url-pattern>/hello</url-pattern>
<!--             /      在服务器解析时,表示地址为http://ip:port/工程路径-->
<!--             工程路径即Deployment配置的路径-->
<!--             /hello 表示地址为 http://ip:port/工程路径/hello-->
         </servlet-mapping>
</web-app>

快捷键:alt+insert 实现所有的方法
设置欢迎页面
在这里插入图片描述

url地址到Servlet程序的访问

在这里插入图片描述

Servlet的生命周期

1.执行Servlet构造器方法
2.执行init初始化方法
3.执行service方法
4.执行destory销毁方法
第1,2步是在第一次访问的时候创建Servlet程序会调用
第3步每次访问都会调用
第4步在web工程停止的时候才会执行
在这里插入图片描述
在这里插入图片描述

package com.Servlet;

import javax.servlet.*;
import java.io.IOException;

public class HelloServlet implements Servlet {
    public HelloServlet() {
        System.out.println("1.构造器方法");
    }

    @Override
    public void init(ServletConfig servletConfig) throws ServletException {
        System.out.println("2.init初始化方法");
    }

    @Override
    public ServletConfig getServletConfig() {
        return null;
    }

    @Override
    /*Servlet程序是专门用来处理请求和响应的*/
    public void service(ServletRequest servletRequest, ServletResponse servletResponse) throws ServletException, IOException {
        System.out.println("3.HelloServlet被访问了");
    }

    @Override
    public String getServletInfo() {
        return null;
    }

    @Override
    public void destroy() {
        System.out.println("4.销毁destory方法");
    }
}

get set请求的分发处理

package com.Servlet;

import javax.servlet.*;
import javax.servlet.http.HttpServletRequest;
import java.io.IOException;

public class HelloServlet implements Servlet {
    public HelloServlet() {
        System.out.println("1.构造器方法");
    }

    @Override
    public void init(ServletConfig servletConfig) throws ServletException {
        System.out.println("2.init初始化方法");
    }

    @Override
    public ServletConfig getServletConfig() {
        return null;
    }

    @Override
    /*Servlet程序是专门用来处理请求和响应的*/
    public void service(ServletRequest servletRequest, ServletResponse servletResponse) throws ServletException, IOException {
        System.out.println("3.HelloServlet被访问了");
        //servletRequest.getmethod();不能用,需要使用其子类型HttpServletRequest,并对其进行强制类型转换
        HttpServletRequest httpServletRequest=(HttpServletRequest) servletRequest;
        String method = httpServletRequest.getMethod();//获取请求的方式get/post
        //System.out.println(method);
        if("GET".equals(method)){
            doGet();

        }else if("POST".equals(method)){
            doPost();
        }
    }
    //一个service方法既要get也要post,有2个任务,这样不太好,应该把他们两个写成两个单独的方法
    //进行请求的分发处理
    //做get请求的操作
    public void doGet(){
        System.out.println("get");
    }
    //做post请求的操作
    public void doPost(){
        System.out.println("post");
    }
    
    
    

    @Override
    public String getServletInfo() {
        return null;
    }

    @Override
    public void destroy() {
        System.out.println("4.销毁destory方法");
    }
}

通过继承HttpServlet实现Servlet程序

一般在实际项目开发中,都是使用继承HttpServlet类的方法去实现Servlet程序
1.编写一个类继承HttpServlet类
2.根据业务需要重写doGet或doPost方法
3.到web.xml中的配置Servlet程序的访问地址

package com.Servlet;

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

public class HelloServlet extends HttpServlet{
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        super.doGet(req, resp);
    }

    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        super.doPost(req, resp);
    }
}

使用IDEA创建Servlet程序

在包里new一个servlet,并取消注解(xml会自动配好)

Servlet类的继承体系

在这里插入图片描述

ServletConfig类

Servlet程序的配置信息类
Servlet 程序和 ServletConfig 对象都是由 Tomcat 负责创建,我们负责使用。
Servlet 程序默认是第一次访问的时候创建,ServletConfig 是每个 Servlet 程序创建时,就创建一个对应的 ServletConfig对象
可以在init方法和doget(getServletConfig)中使用

三大作用

1.可以获取Servlet程序的别名servlet-name的值
2.获取初始化参数init-param

             <init-param>
<!--                 设置参数名-->
                 <param-name>username</param-name>
<!--                 设置参数值-->
                 <param-value>root</param-value>
             </init-param>

3.获取ServletContext对象

public void init(ServletConfig config) throws ServletException {
        System.out.println("init初始化方法");
        System.out.println("Servlet的别名是"+config.getServletName());
        System.out.println("获取初始化参数"+config.getInitParameter("username"));
        System.out.println("获取初始化参数"+config.getInitParameter("url"));

        System.out.println("获取ServletContet对象"+config.getServletContext());

    }

在这里插入图片描述

ServletContext类

1、ServletContext 是一个接口,它表示 Servlet 上下文对象
2、一个 web 工程,只有一个 ServletContext 对象实例。
3、ServletContext 对象是一个域对象
4、ServletContext 是在 web 工程部署启动的时候创建。在 web 工程停止的时候销毁。
什么是域对象?
域对象,是可以像 Map 一样存取数据的对象,叫域对象。
这里的域指的是存取数据的操作范围:整个 web 工程
在这里插入图片描述

ServletContext 类的四个作用

1、获取 web.xml 中配置的上下文参数 context-param

<!--context-param是上下文参数(它属于整个web工程)-->
        <context-param>
            <param-name>username</param-name>
            <param-value>ddd</param-value>
        </context-param>
        <context-param>
            <param-name>password</param-name>
            <param-value>111</param-value>
        </context-param>

注意init-param只能由ServletConfig获得
context-param只能由ServletContext对象获得

2、获取当前的工程路径,格式: /工程路径
3、获取工程部署后在服务器硬盘上的绝对路径
4、像 Map 一样存取数据

//Servlet1
context.setAttribute("key1", "value1"); 
System.out.println("Context1 中获取域数据 key1 的值是:"+ context.getAttribute("key1"));

Servlet2也可以获取到Servlet1中的Attribute
ServletContext 是在 web 工程部署启动的时候创建。在 web 工程停止的时候销毁,里面的Attribute也随之销毁

public void init(ServletConfig config) throws ServletException {
        //重写了init方法一定要调用
        super.init(config);
        // 1、获取 web.xml 中配置的上下文参数context-param
        ServletContext context = getServletConfig().getServletContext();
        String username = context.getInitParameter("username");
        System.out.println("context-param 参数 username 的值是:" + username);
        System.out.println("context-param 参数 password 的值是:" + context.getInitParameter("password"));
        // 2、获取当前的工程路径,格式: /工程路径
        System.out.println( "当前工程路径:" + context.getContextPath() );
        // 3、获取工程部署后在服务器硬盘上的绝对路径 
        /*
        / 斜杠被服务器解析地址为:http://ip:port/工程名/ 
        映射到 IDEA 代码的 web 目录 */
        System.out.println("工程部署的路径是:" + context.getRealPath("/"));
        
        System.out.println("工程下 css 目录的绝对路径是:" + context.getRealPath("/css"));
        System.out.println("工程下 imgs 目录 1.jpg 的绝对路径是:" + context.getRealPath("/imgs/1.jpg"));

    }

输出:

context-param 参数 username 的值是:
ddd
context-param 参数 password 的值是:
111
当前工程路径:
/07_Servlet
工程部署的路径是:
C:\Tomcat\apache-tomcat-8.0.50-windows-x64\apache-tomcat-8.0.50\webapps\07_Servlet
工程下 css 目录的绝对路径是:
C:\Tomcat\apache-tomcat-8.0.50-windows-x64\apache-tomcat-8.0.50\webapps\07_Servlet\css
工程下 imgs 目录 1.jpg 的绝对路径是:
C:\Tomcat\apache-tomcat-8.0.50-windows-x64\apache-tomcat-8.0.50\webapps\07_Servlet\imgs\1.jpg

IDEA整合Tomcat之后,Tomcat被拷贝的副本内容
IDEA整合tomcat就是在catalina/local下创建xml配置文件部署web工程
Using CATALINA_BASE:
“C:\Users\ddd\AppData\Local\JetBrains\IntelliJIdea2021.1\tomcat\f907c3a6-a283-4453-868f-b2e6e1e0f712”

HTTP协议

什么是协议?

协议是指双方,或多方,相互约定好,大家都需要遵守的规则,叫协议。 所谓 HTTP 协议,就是指,客户端和服务器之间通信时,发送的数据,需要遵守的规则,叫 HTTP 协议。 HTTP 协议中的数据又叫报文

请求的 HTTP 协议格式

客户端给服务器发送数据叫请求。
服务器给客户端回传数据叫响应。
请求又分为 GET 请求,和 POST 请求两种

GET 请求

1、请求行
(1) 请求的方式 GET
(2) 请求的资源路径[+?+请求参数]
(3) 请求的协议的版本号 HTTP/1.1
2、请求头
key : value 组成 不同的键值对,表示不同的含义。
在这里插入图片描述

ii. POST 请求

1、请求行
(1) 请求的方式 POST
(2) 请求的资源路径[+?+请求参数]
(3) 请求的协议的版本号 HTTP/1.1
2、请求头

  1. key : value 不同的请求头,有不同的含义

---------------空行 ---------------

3、请求体
===>>> 就是发送给服务器的数据
在这里插入图片描述

常用请求头的说明

Accept: 表示客户端可以接收的数据类型
Accpet-Languege: 表示客户端可以接收的语言类型
User-Agent: 表示客户端浏览器的信息
Host: 表示请求时的服务器 ip 和端口号

哪些是 GET 请求,哪些是 POST 请求

GET 请求有哪些:
1、form 标签 method=get
2、a 标签
3、link 标签引入 css
4、Script 标签引入 js 文件
5、img 标签引入图片
6、iframe 引入 html 页面
7、在浏览器地址栏中输入地址后敲回车

POST 请求有哪些:
8、form 标签 method=post

响应的 HTTP 协议格式

1、响应行
(1) 响应的协议和版本号 HTTP/1.1
(2) 响应状态码 200
(3) 响应状态描述符 ok
2、响应头
(1) key : value 不同的响应头,有其不同含义
------------------空行---------------------------
3、响应体
---->>> 就是回传给客户端的数据
在这里插入图片描述

常用的响应码说明

200 表示请求成功
302 表示请求重定向
404 表示请求服务器已经收到了,但是你要的数据不存在(请求地址错误)
500 表示服务器已经收到请求,但是服务器内部错误(代码错误)

MIME 类型说明

MIME 是 HTTP 协议中数据类型。
MIME 的英文全称是"Multipurpose Internet Mail Extensions" ,多功能 Internet 邮件扩充服务
MIME 类型的格式是“大类型/小类型”,并与某一种文件的扩展名相对应。
常见的 MIME 类型:
在这里插入图片描述
在这里插入图片描述

F12+Network
详细内容点在这里插入图片描述
响应体在Response

在这里插入图片描述

HttpServletRequest类

作用

每次只要有请求进入 Tomcat 服务器,Tomcat 服务器就会把请求过来的 HTTP 协议信息解析好封装到 Request 对象中。
然后传递到 service 方法(doGet 和 doPost)中给我们使用。我们可以通过 HttpServletRequest 对象,获取到所有请求的信息。

i. getRequestURI() 获取请求的资源路径
ii. getRequestURL() 获取请求的统一资源定位符(绝对路径
iii. getRemoteHost() 获取客户端的 ip 地址
iv. getHeader() 获取请求头
v. getParameter() 获取请求的参数
vi. getParameterValues() 获取请求的参数(多个值的时候使用
vii. getMethod() 获取请求的方式 GET 或 POST
viii. setAttribute( key, value) 设置域数据
ix. getAttribute (key) 获取域数据
x. getRequestDispatcher() 获取请求转发对象

protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        super.doGet(req, resp);
        // i.getRequestURI() 获取请求的资源路径
        System.out.println("URI => " + req.getRequestURI());
        // ii.getRequestURL() 获取请求的统一资源定位符(绝对路径)
        System.out.println("URL => " + req.getRequestURL());
        // iii.getRemotest() 获取客户端的 ip 地址
        // * 在 IDEA 中,使用 localhost 访问时,得到的客户端 ip 地址是 ===>>> 127.0.0.1
        // * 在 IDEA 中,使用 127.0.0.1 访问时,得到的客户端 ip 地址是 ===>>> 127.0.0.1
        // * 在 IDEA 中,使用 真实 ip 访问时,得到的客户端 ip 地址是 ===>>> 真实的客户端 ip地址
        System.out.println("客户端 ip 地址 => " + req.getRemoteHost());
        // iv.getHeader() 获取请求头
        System.out.println("请求头 User-Agent ==>> " + req.getHeader("User-Agent"));
        // vii.getMethod() 获取请求的方式 GET 或 POST
        System.out.println( "请求的方式 ==>> " +req.getMethod() );
    }

在这里插入图片描述
在这里插入图片描述

如何获取请求参数

<body> 
<form action="http://localhost:8080/07_Servlet/hello" method="get"> 
    用户名:<input type="text" name="username"><br/> 
    密码:<input type="password" name="password"><br/> 
    兴趣爱好:
    <input type="checkbox" name="hobby" value="cpp">C++ 
    <input type="checkbox" name="hobby" value="java">Java 
    <input type="checkbox" name="hobby" value="js">JavaScript<br/> 
    <input type="submit"> 
</form> 
</body>
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {

        String username = req.getParameter("username");
        String password = req.getParameter("password");
        String[] hobby = req.getParameterValues("hobby");
        System.out.println("用户名:" + username);
        System.out.println("密码:" + password);
        System.out.println("兴趣爱好:" + Arrays.asList(hobby));
    }

doGet 请求的中文乱码解决

// 获取请求参数 
String username = req.getParameter("username"); 
//1 先以 iso8859-1 进行编码 
//2 再以 utf-8 进行解码 
username = new String(username.getBytes("iso-8859-1"), "UTF-8");

POST 请求的中文乱码解决

@Override 
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        // 设置请求体的字符集为 UTF-8,从而解决 post 请求的中文乱码问题  
        req.setCharacterEncoding("UTF-8"); 
        System.out.println("-------------doPost------------");
        // 获取请求参数 
        String username = req.getParameter("username"); 
        String password = req.getParameter("password");
        String[] hobby = req.getParameterValues("hobby"); System.out.println("用户名:" + username);
        System.out.println("密码:" + password);
        System.out.println("兴趣爱好:" + Arrays.asList(hobby));
    }

请求的转发

定义:服务器收到请求后,从一个资源跳转到另一个资源的操作

运行在服务器上的两个Servlet程序共同完成一个完整业务功能,业务的执行有顺序
在这里插入图片描述

请求转发的特点
1.浏览器地址栏没有变化
2.他们是一次请求
3.他们共享request域中的数据
4.可以转发到WEB-INF目录下,浏览器不能访问
/WEB-INF/form.html
5.不可以访问工程以外的资源
eg.不能跳到百度

hello

protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {

        // 获取请求的参数(办事的材料)查看
        String username = req.getParameter("username");
        System.out.println("在 Servlet1(柜台 1)中查看参数(材料):" + username);
        // 给材料 盖一个章,并传递到 Servlet2(柜台 2)去查看,
        req.setAttribute("key1","柜台 1 的章");
        // 问路:Servlet2(柜台 2)怎么走
        //请求转发必须要以斜杠打头,
        // / 斜杠表示地址为:http://ip:port/工程名/ , 映射到 IDEA 代码的 web 目录
        RequestDispatcher requestDispatcher = req.getRequestDispatcher("/hello1");
        //RequestDispatcher requestDispatcher = req.getRequestDispatcher("http://www.baidu.com");
        // 走向 Sevlet2(柜台 2)
        requestDispatcher.forward(req,resp);
    }

hello1

protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        // 获取请求的参数(办事的材料)查看
        String username = req.getParameter("username");
        System.out.println("在 Servlet2(柜台 2)中查看参数(材料):" + username);
        // 查看 柜台 1 是否有盖章
        Object key1 = req.getAttribute("key1");
        System.out.println("柜台 1 是否有章:" + key1);
        // 处理自己的业务
        System.out.println("Servlet2 处理自己的业务 ");
    }

在这里插入图片描述

base标签的作用

07_Servlet/a/b/c.html

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
    <!--<base href="http://localhost:8080/07_Servlet/a/b/">-->
</head>
<body>
     <a href="../../index.html">跳回首页</a>
</body>
</html>

07_Servlet/index.html

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>首页</title>
</head>
<body>
     这是Web下的index.html<br/>
     <a href="/a/b/c.html" >a/b/c.html</a></br>
     <a href="http://localhost:8080/07_Servlet/hello">请求转发:a/b/c.html</a>
</body>
</html>

在这里插入图片描述

<!DOCTYPE html> <html lang="zh_CN"> 
<head> 
      <meta charset="UTF-8"> 
      <title>Title</title> 
      <!--base 标签设置页面相对路径工作时参照的地址 
      href 属性就是参数的地址值 --> 
      <base href="http://localhost:8080/07_servlet/a/b/c.html"> 
      <!--c.html可以省略,但/不行,http://localhost:8080/07_servlet/a/b/-->
</head> 
<body> 
      这是 a 下的 b 下的 c.html 页面<br/> 
     <a href="../../index.html">跳回首页</a><br/>
</body> 
</html>

Web 中的相对路径和绝对路径

在 javaWeb 中,路径分为相对路径和绝对路径两种:
相对路径

. 表示当前目录
… 表示上一级目录
资源名 表示当前目录/资源名

绝对路径

http://ip:port/工程路径/资源路径
在实际开发中,路径都使用绝对路径,而不简单的使用相对路径。
1、绝对路径
2、base+相对

web 中 / 斜杠的不同意义

在 web 中 / 斜杠 是一种绝对路径
/ 斜杠 如果被浏览器解析,得到的地址是:http://ip:port/

 <a href="/">斜杠</a>

/ 斜杠 如果被服务器解析,得到的地址是:http://ip:port/工程路径

1、<url-pattern> /servlet1</url-pattern>
2、servletContext.getRealPath(“/”);
3、request.getRequestDispatcher(“/”);

特殊情况
response.sendRediect(“/”); 请求重定向
把斜杠发送给浏览器解析。得到 http://ip:port/

HttpServletResponse 类

HttpServletResponse 类的作用

HttpServletResponse 类和 HttpServletRequest 类一样。
每次请求进来,Tomcat 服务器都会创建一个 Response 对象传递给Servlet 程序去使用。
HttpServletRequest 表示请求过来的信息,HttpServletResponse 表示所有响应的信息, 我们如果需要设置返回给客户端的信息,都可以通过 HttpServletResponse 对象来进行设置

两个输出流的说明

字节流 getOutputStream()
常用于下载(传递二进制数据)
字符流 getWriter()
常用于回传字符串(常用)

两个流同时只能使用一个。 使用了字节流,就不能再使用字符流,反之亦然,否则就会报错。

如何往客户端回传数据

要求 : 往客户端回传 字符串 数据

public class ResponseIOServlet extends HttpServlet { 
@Override 
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { 
// 要求 : 往客户端回传 字符串 数据。 
             PrintWriter writer = resp.getWriter(); 
             writer.write("response's content!!!"); 
     } 
}

响应的乱码解决

解决响应中文乱码方案一(不推荐使用):

// 设置服务器字符集为 UTF-8 
resp.setCharacterEncoding("UTF-8"); 
// 通过响应头,设置浏览器也使用 UTF-8 字符集 
resp.setHeader("Content-Type", "text/html; charset=UTF-8");

解决响应中文乱码方案二(推荐):

// 它会同时设置服务器和客户端都使用 UTF-8 字符集,还设置了响应头 
// 此方法一定要在获取流对象之前调用才有效 
resp.setContentType("text/html; charset=UTF-8");

请求重定向

请求重定向,是指客户端给服务器发请求,然后服务器告诉客户端说:我给你一些地址,你去新地址访问。叫请求重定向(因为之前的地址可能已经被废弃)
在这里插入图片描述

请求重定向的特点:

1.浏览器地址栏会发生变化
2.两次请求
3.不共享Request域中的数据
两次请求,获得的是不同的Request对象
4.不能访问WEB-INF下的资源
WEB-INF浏览器无法直接访问
5.可以访问工程外的资源

请求重定向的第一种方案:

// 设置响应状态码 302 ,表示重定向,(已搬迁) resp.setStatus(302); 
// 设置响应头,说明 新的地址在哪里 
resp.setHeader("Location", "http://localhost:8080/07_Servlet/hello"); 

请求重定向的第二种方案(推荐使用):

resp.sendRedirect("http://localhost:8080/Servlet/hello");

请求转发和重定向总结

1.转发使用的是getRequestDispatcher()方法;
重定向使用的是sendRedirect();
2.转发:浏览器URL的地址栏不变。重定向:浏览器URL的地址栏改变;
3.转发是服务器行为,重定向是客户端行为;
4.转发是浏览器只做了一次访问请求。重定向是浏览器做了至少两次的访问请求;
5.转发2次跳转之间传输的信息不会丢失,重定向2次跳转之间传输的信息会丢失(request范围);

转发和重定向的选择

重定向的速度比转发慢,因为浏览器还得发出一个新的请求,如果在使用转发和重定向都无所谓的时候建议使用转发。
因为转发只能访问当前WEB的应用程序,所以不同WEB应用程序之间的访问,特别是要访问到另外一个WEB站点上的资源的情况,这个时候就只能使用重定向了。

转发和重定向的应用场景

转发是要比重定向快,因为重定向需要经过客户端,而转发没有。有时候,采用重定向会更好,若需要重定向到另外一个外部网站,则无法使用转发。另外,重定向还有一个应用场景:避免在用户重新加载页面时两次调用相同的动作

例如,当提交产品表单的时候,执行保存的方法将会被调用,并执行相应的动作;这在一个真实的应用程序中,很有可能将表单中的所有产品信息加入到数据库中。但是如果在提交表单后,重新加载页面,执行保存的方法就很有可能再次被调用。同样的产品信息就将可能再次被添加,为了避免这种情况,提交表单后,你可以将用户重定向到一个不同的页面,这样的话,这个网页任意重新加载都没有副作用;

但是,使用重定向不太方便的地方是,使用它无法将值轻松地传递给目标页面。而采用转发,则可以简单地将属性添加到Model,使得目标视图可以轻松访问。由于重定向经过客户端,所以Model中的一切都会在重定向时丢失。但幸运的是,在Spring3.1版本以后,我们可以通过Flash属性,解决重定向时传值丢失的问题。

要使用Flash属性,必须在Spring MVC的配置文件中添加一个。然后,还必须再方法上添加一个新的参数类型:org.springframework.web.servlet.mvc.support.RedirectAttributes。

  Java知识库 最新文章
计算距离春节还有多长时间
系统开发系列 之WebService(spring框架+ma
springBoot+Cache(自定义有效时间配置)
SpringBoot整合mybatis实现增删改查、分页查
spring教程
SpringBoot+Vue实现美食交流网站的设计与实
虚拟机内存结构以及虚拟机中销毁和新建对象
SpringMVC---原理
小李同学: Java如何按多个字段分组
打印票据--java
上一篇文章      下一篇文章      查看所有文章
加:2021-11-17 12:37:37  更:2021-11-17 12:38:13 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/24 2:02:45-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码