请求转发和重定向

request除了可以作为请求对象之外,还可以作为域对象,但是该域对象的取值范围,是一次请求范围之内(浏览器地址栏没有发生跳转访问别的资源)

作用:将servlet中的数据通过request对象带到jsp页面;

请求转发

  1. 通过请求对象获取请求转发对象

    RequestDispatcher requestDispatcher = req.getRequestDispatcher("资源相对路径");
  2. 定位具体资源

    requestDispatcher.forward(req,resp);

实现内部资源的跳转:通过浏览器发起访问,工程中的资源会从一个地址跳转到另外一个地址,但浏览器端的地址不会发生改变。

请求对象req还可以作为与对象存放数据,因为请求转发方式是不会发生地址的跳转的,所以在req的作用范围(一次请求范围)之内,因此可以在跳转的资源中获取到存放的数据。

@WebServlet("/hello01")
public class HelloServlet extends HttpServlet { @Override
protected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { //往域中存放数据
req.setAttribute("username","jack");
req.setAttribute("password","123456"); //获取请求转发对象 参数是另一个资源的相对路径
RequestDispatcher requestDispatcher = req.getRequestDispatcher("/hello02"); //调用forward 定位具体资源
requestDispatcher.forward(req,resp); }
}
@WebServlet("/hello02")
public class HelloServlet02 extends HttpServlet { @Override
protected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { //取出域中的数据
Object username = req.getAttribute("username");
Object password = req.getAttribute("password");
System.out.println(username);
System.out.println(password); }
}

输出结果:

jack
123456

在访问这个地址后,地址栏的地址没有发生跳转,但是访问的资源已经到了/hello02,通过后端控制台打印就可看到获取到了数据。

请求重定向

可以完成内部资源的跳转,而且浏览器的地址是会发生改变的,这种情况下,不能取到request域中的数据

hello03后端资源:

@WebServlet("/hello03")
public class HelloServlet03 extends HttpServlet { @Override
protected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { //往域中存放数据
req.setAttribute("key","value"); //请求重定向 是响应对象
//参数是绝对路径
resp.sendRedirect("http://localhost:8080/servlet/hello04"); }
}

hello04后端资源:

@WebServlet("/hello04")
public class HelloServlet04 extends HttpServlet { @Override
protected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { //取出数据
Object key = req.getAttribute("key");
System.out.println("----"+key+"----"); }
}
  1. 通过浏览器方法hello03资源

  2. 浏览器端地址跳转

  3. 控制台打印key值

    因为发生了重定向是要发生地址跳转的,超出了req与对象的作用范围(一次请求)内,所以在hello04资源中是获取不到hello03资源的req的数据的。

请求重定向到静态html页面

hello05:

@WebServlet("/hello05")
public class HelloServlet05 extends HttpServlet { @Override
protected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { /*//获取转发对象 相对路径 前端资源/代表web文件夹下
RequestDispatcher requestDispatcher = req.getRequestDispatcher("/html/test.html"); //定位到具体资源
requestDispatcher.forward(req,resp);*/ //请求重定向
resp.sendRedirect("http://localhost:8080/servlet/html/test.html"); }
}

test.html:

<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>test</title>
</head>
<body>
成功访问了前端静态资源 </body>
</html>

浏览器访问结果:

总结:

存放在ServletContext域对象中的数据,不管是通过重定向还是请求转发都是可以在跳转的资源中的获取到的。

存放在请求域对象request中的数据,如果是通过请求转发,没有地址的跳转,是可以在另一资源中获取到的;相反,若干是重定向,有地址的跳转,那么不可以在另一个资源中获取到访问资源request中存放的数据。

JSP模板引擎

html是前端中静态的资源,是不能直接获取到后端的资源的。jsp模板引擎,可以动态渲染后端数据到前端,jsp中的el表达式可以取出后端域中的数据。

格式:

${域对象的key}
@WebServlet("/hello06")
public class HelloServlet06 extends HttpServlet { @Override
protected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { User user1=new User("jack1","132");
User user2=new User("jack2","132");
User user3=new User("jack3","132");
ArrayList<User> users = new ArrayList<>();
users.add(user1);
users.add(user2);
users.add(user3); //获取ServletContext域对象
ServletContext servletContext = getServletContext();
servletContext.setAttribute("users",users); //重定向
resp.sendRedirect("http://localhost:8080/servlet/jsp/test.jsp");
}
}

test.jsp:

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

浏览器访问结果:

重定向有地址的跳转

JSP

JSP(全称JavaServer Pages)是由Sun Microsystems公司主导创建的一种动态网页技术标准。JSP部署于网络服务器上,可以响应客户端发送的请求,并根据请求内容动态地生成HTML、XML或其他格式文档的Web网页,然后返回给请求者。JSP技术以Java语言作为脚本语言,为用户的HTTP请求提供服务,并能与服务器上的其它Java程序共同处理复杂的业务需求。JSP将Java代码和特定变动内容嵌入到静态的页面中,实现以静态页面为模板,动态生成其中的部分内容。

JSP是一个前端的模板引擎,用于渲染后端数据,底层本质是一个Servlet程序。JSP可以动态的获取到后端的数据,由于底层是servlet,servlet底层是java,所以可以在代码脚本中去编写java代码

EL表达式

可以在后端资源中的域对象ServletContext中设置属性,然后在跳转的JSP文件中通过EL表达式获取到属性值。可以用request请求域对象和ServletContext域对象来设置属性和值,这基于方式是请求转发方式还是请求重定向方式,如果是请求转发方式,那么可以用request请求域对象或者ServletContext域对象来setAttribute,因为请求转发方式不会发生地址的跳转,request请求域对象也可以进行传递值;如果是重定向方式,那么就只能用ServletContext域对象来setAttribute,因为其作用范围广,地址跳转,仍然能够存取数据;

1、重定向方式

case2Servlet

@WebServlet("/case2Servlet")public class case2Servlet extends HttpServlet {    @Override    protected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {        resp.setContentType("text/html;charset=UTF-8");        ArrayList<User> users = Jdbc.selectDB();        //采用重定向方式        ServletContext servletContext = getServletContext();        servletContext.setAttribute("users",users);        resp.sendRedirect("http://localhost:8080/servlet/jsp/case2Jsp01.jsp");    }}

case2Jsp01.jsp

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

浏览器访问:

2、请求转发方式

JSP脚本

  • 声明脚本的格式:<%!声明java代码%>
  • 表达式脚本的格式:<%=表达式%>
  • 代码脚本的格式:<%java语句%>
<%@ page import="java.util.ArrayList" %><%@ page contentType="text/html;charset=UTF-8" language="java" %><%--jsp代码脚本--%><%    String name="Tom";    Integer age=50;    Object o=new Object();    System.out.println(name+" "+age+" "+o);    for (int i=0;i<10;i++)        System.out.println(i);    ArrayList<Object> objects = new ArrayList<>();//会自动导包	ArrayList<User> users = new ArrayList<>();    users.add(new User(12,"jack1","123456"));    users.add(new User(12,"jack2","123456"));    users.add(new User(12,"jack3","123456"));    users.add(new User(12,"jack4","123456"));%><%--声明脚本--%><%!    private Integer i=5;%><html><head>    <title>test</title></head><body>    <%--表达式脚本--%>    <%=age%>    <%=name%>    <%=users.get(0).getUsername()%></body></html>

JSP九大内置对象:

JSP程序在执行的时候,底层会生成对应的源代码和字节码文件,在源代码中,会封装9大内置对象,包括字符流对象、异常对象、域对象

  1. request
  2. response
  3. pageContext
  4. exception
  5. application
  6. out
  7. page
  8. session
  9. servletConfig

JSP四大域对象:

域对象是都可以存取值的

  1. pageContext

    JSP内置的文本域对象,取值范围是当前的JSP页面,出了该页面的数据就无法访问

  2. request

    是一个请求域对象,取值范围是一次请求范围之类,浏览器只要地址不发生改变,则可以访问到数据

  3. session

    是一个会话域对象,取值范围是浏览器的开启到关闭,对应的生命周期也是浏览器的开启到关闭

  4. application

    实际上就是一个ServletContext对象,取值范围是整个web工程

下面根据代码的栗子来说明不同的取值范围:

  1. 首先在test04.jsp文件 JSP代码脚本各种域对象存放数据,然后在body部分进行取出数据

test04.jsp

<%@ page contentType="text/html;charset=UTF-8" language="java" %><%--声明一个jsp脚本--%><%    //存放数据到域中    request.setAttribute("key1","value1");    request.setAttribute("key2","value2");    request.setAttribute("key3","value3");    pageContext.setAttribute("pageContext","pageContext");    request.setAttribute("request","request");    session.setAttribute("session","session");    application.setAttribute("application","application");%><html><head>    <title>test</title></head><body>    <%--用表达式脚本取出域中的数据--%>    <%=request.getAttribute("key1")%><br>    <%=request.getAttribute("key2")%><br>    <%=request.getAttribute("key3")%><br>    pageContext的value值为:<%=pageContext.getAttribute("pageContext")%><br>    request的value值为:<%=request.getAttribute("request")%><br>    session的value值为:<%=session.getAttribute("session")%><br>    application的value值为:<%=application.getAttribute("application")%><br></body></html>
  1. 浏览器端

​ 可以看到4个域对象现在都能访问到其中存取的数据

  1. 在同一工程下新建一个test05.jsp,用于取出域对象中的数据
<%@ page contentType="text/html;charset=UTF-8" language="java" %><html><head>    <title>test</title></head><body>    pageContext的value值为:<%=request.getAttribute("pageContext")%><br>    request的value值为:<%=request.getAttribute("request")%><br>    session的value值为:<%=session.getAttribute("session")%><br>    application的value值为:<%=application.getAttribute("application")%><br></body></html>
  1. 浏览器端访问

​ 可以发现pageContext和request域对象的值已经获取不到了,因为pageContext的取值范围是当前JSP文件,request的取值范围是一次请求范围之内。访问的JSP文件改变,访问地址改变,明显超出了这两者的取值范围。

  1. 关闭浏览器,重新打开浏览器,再次访问test05.jsp页面

    可以发现session域对象的值也获取不到了,因为session域对象的取值范围是浏览器的开启到关闭。而application是一直可以获取到的,因为它的取值范围是整个web工程,只要web工程运行,就可以随时在任何地方都能访问到值。

JavaWeb 请求转发重定向的更多相关文章

  1. 20160326 javaweb 请求转发和请求包含

    (1)请求转发: this.getServletContext().getRequestDispatcher("").forward(request,response); requ ...

  2. JavaWeb中请求转发和请求重定向的区别

    针对于JavaWeb中请求与重定向的一个cheatsheep: 1.转发 1)完成一次转发,用户浏览器发送一次请求 2)转发之后,浏览器URL地址栏不改变(服务器帮忙完成) 3)请求域中数据不丢失 4 ...

  3. javaweb之Servlet,http协议以及请求转发和重定向

    本文是作者原创,版权归作者所有.若要转载,请注明出处. 一直用的框架开发,快连Servlet都忘了,此文旨在帮自己和大家回忆一下Servlet主要知识点.话不多说开始吧 用idea构建Servlet项 ...

  4. Java 请求转发和重定向的区别以及JavaWeb三大作用域

    三大作用域以及转发和重定向 学习总结 1. 转发和重定向 转发 重定向 转发和重定向的区别: 什么时候用转发什么时候用重定向 三大作用域 作用域类型 作用域方法 如何选择作用域 总结 学习总结 1. ...

  5. javaweb中重定向和请求转发(response.sendRedirect()和request.getRequestDispatcher(rul).forward(request,response)))的区别

    先来两张图,方便理解: 可以看出,重定向时,是服务器向游览器重新发送了一个response命令,让游览器再次向url2发送请求,以获取url2的资源 而请求转发时,类似于是服务器自己向自己发了一个跳转 ...

  6. javaWeb中request请求转发和response重定向

    1.访问资源 运用forward方法只能重定向到同一个Web应用程序中的一个资源. 而sendRedirect方法可以让你重定向到任何URL.  2.request.get Forward代码中的&q ...

  7. ServletRequest HttpServletRequest 请求方法 获取请求参数 请求转发 请求包含 请求转发与重定向区别 获取请求头字段

      ServletRequest 基本概念 JavaWeb中的 "Request"对象  实际为   HttpServletRequest  或者  ServletRequest, ...

  8. HTTP协议基础与web服务的重定向,跳转以及请求转发

    JavaWeb中,HttpServletRequest与HttpServletResponse几乎是处理各种请求与操作必备的参数,与原始的ServletRequest/ServletResponse相 ...

  9. spring mvc 请求转发和重定向(转)

    spring mvc controller间跳转 重定向 传参 url:http://zghbwjl.blog.163.com/blog/static/12033667220137795252845/ ...

随机推荐

  1. [源码解析]PyTorch如何实现前向传播(2) --- 基础类(下)

    [源码解析]PyTorch如何实现前向传播(2) --- 基础类(下) 目录 [源码解析]PyTorch如何实现前向传播(2) --- 基础类(下) 0x00 摘要 0x01 前文回顾 0x02 Te ...

  2. Go语言核心36讲(Go语言进阶技术三)--学习笔记

    09 | 字典的操作和约束 至今为止,我们讲过的集合类的高级数据类型都属于针对单一元素的容器. 它们或用连续存储,或用互存指针的方式收纳元素,这里的每个元素都代表了一个从属某一类型的独立值. 我们今天 ...

  3. docker内服务访问宿主机服务

    目录 1. 场景 2. 解决 4. 参考 1. 场景 使用windows, wsl2 进行日常开发测试工作. 但是wsl2经常会遇到网络问题.比如今天在测试一个项目,核心功能是将postgres 的数 ...

  4. 【UE4 调试】C++ 几种编译方法和小技巧

    编译方法 Visual Studio 2019 编译 默认编译 UnrealVS 快速编译 Editor 编译 一般 vs 编译完后,Editor会跟着热编译(有声音) 如果发现编译后代码没更新到Ed ...

  5. Java:并发笔记-06

    Java:并发笔记-06 说明:这是看了 bilibili 上 黑马程序员 的课程 java并发编程 后做的笔记 5. 共享模型之无锁 本章内容 CAS 与 volatile 原子整数 原子引用 原子 ...

  6. Java:并发笔记-02

    Java:并发笔记-02 说明:这是看了 bilibili 上 黑马程序员 的课程 java并发编程 后做的笔记 3. 共享模型之管程-1 本章内容-1 共享问题 synchronized 线程安全分 ...

  7. 防止SQL注入总结

    1.预编译(占位符)可以很大程度上防止SQL注入 预编译的原理是数据库厂商提供的JAR包中,对参数进行了转义 2.mybatis中,能用# 的地方,不用$,因为#是预编译占位符形式,可以防止SQL注入 ...

  8. 表单编辑时el-form的validate方法执行无效,阻塞代码运行 - Element UI踩坑记录

    今天在用element-ui写管理后台需求时,遇到一个奇怪的问题 一个正常带校验的表单,在新增列表数据时表单校验功能正常: 但是在新增之后再去编辑数据时,表单校验却失效了,甚至阻塞了后续的代码执行,控 ...

  9. GT考试

    比较神仙的$dp+KMP+Matrix$综合题目,比较值得一写 $0x00$:首先我打了一个爆搜 不过对正解并无任何启发...(逗比发言请忽略) $0x01$:基础$dp$ 状态还是比较好设的, 考虑 ...

  10. 对SQLServer错误使用聚集索引的优化案例(千万级数据量)

    前言: 半个月前发了文章 SQLServer聚集索引导致的插入性能低 终于等到生产环境休整半天,这篇文章是对前文的实际操作. 以下正文开始: 异常:近期发现偶尔有新数据插入超时. 分析:插入条码有多种 ...