同样是“跳转”,为何forward地址栏不变,redirect会变?

故事场景:在大型办公楼里办业务

你(浏览器/用户 )今天要去"市政府办公大楼"(服务器 )办理一项业务。你手裡拿著一份填好的"申请表 "(HttpServletRequest)。

方式一:forward --- "内部转交,您请稍候"

这是一种对你透明、在内部完成的服务方式。

  • 流程:

      1. 你来到大楼前台(比如 ForwardServlet),把你的"申请表"递给了接待员A。
      1. 接待员A看了一眼,说:"好的,我受理了。但这个业务需要后台的档案室(FinalDestinationServlet)处理。您在这儿稍等,别动。"
      1. 然后,接待员A拿着你的那份原封不动的申请表 ,甚至可能在上面用便签加了一条批注(request.setAttribute(...)),亲自跑到了大楼内部的档案室,把表交给了档案员B。
      1. 档案员B处理完,将最终结果通过接待员A交还给你。
  • 你的体验:

    • 地址栏不变:从始至终,你都站在前台,你的物理位置没有变过。你手机导航的目的地(浏览器地址栏URL)一直显示的是"市政府前台"。

    • 一次请求:你只来了一次办公大楼。

    • 数据共享 :接待员A在你申请表上加的便签,档案员B能原封不动地看到。因为你们自始至终用的都是同一份申请表

方式二:redirect --- "地址不对,请您再去一趟"

这是一种让你自己跑腿的服务方式。

  • 流程:

      1. 你来到大楼前台(比如 RedirectServlet),把你的"申请表"递给了接待员C。
      1. 接待员C看了一眼,立刻把申请表退还给你,说:"哦!您走错地方了! 这个业务现在已经搬到街对面的'市民服务中心'(/finalDestination)去办了。这是新地址,您得自己过去一趟。"
      1. 接待员C的工作到此结束。他给了你一个"302重定向"指令。
      1. 你只好走出办公大楼,回到车里,重新导航到"市民服务中心"这个新地址 ,然后重新提交一份申请表
  • 你的体验:

    • 地址栏改变 :你 physically 移动到了新的地点,所以你的手机导航(浏览器地址栏URL)也更新成了新的地址"市民服务中心"。

    • 两次请求:你先去了"市政府",又去了"市民服务中心",这是完全独立的两次出行。

    • 数据不共享 :你在第一份申请表上精心粘贴的补充材料,在去新地点的路上弄丢了。因为你去新地点提交的是一份全新的申请表,和旧的那份毫无关系。

故事总结:

特性 forward (内部转交) redirect (请您再去一趟)
发生地点 服务器内部 ,对浏览器透明 服务器 -> 浏览器 -> 服务器
请求次数 1次 2次
地址栏URL 不变 改变
数据共享 共享 request 数据 (效率高) 不共享 request 数据 (效率低)
核心比喻 一个部门把你的材料转给另一个部门 告诉你走错门了,让你自己去另一个部门
本质 服务器帮你处理,你无感知 服务器让你自己去另一个地方
使用场景 MVC框架中,Controller处理完数据后,转发给View(JSP)进行渲染。 用户登录成功后,重定向到主页;处理完表单提交后,重定向到结果页(防止重复提交)。

一句话总结:

  • forward 是"我帮你办",发生在服务器内部,快,能带东西。

  • redirect 是"你自个儿去",发生在浏览器端,慢,两手空空地去。

核心代码

我们将用三个简化的 Servlet 来演示 forwardredirect 的行为差异,特别是它们在"地址栏变化"和"请求数据传递"上的不同。

1. 发起转发的 Servlet (ForwardServlet.java)

复制代码
// ForwardServlet.java
@WebServlet("/forwardServlet")
publicclassForwardServletextendsHttpServlet {
    protectedvoiddoGet(HttpServletRequest request, HttpServletResponse response)throws ServletException, IOException {
        System.out.println("ForwardServlet: 接待了请求,正在处理...");
        
        // forward 可以将数据放入 request 域中,传递给下一个 servlet
        request.setAttribute("message", "这是来自 ForwardServlet 的秘密文件!");
        
        System.out.println("ForwardServlet: 决定将请求 '内部转交' 给 FinalDestinationServlet。");
        
        // 获取请求分发器,并执行 forward
        RequestDispatcherdispatcher= request.getRequestDispatcher("/finalDestination");
        dispatcher.forward(request, response);
        
        // forward 之后,这里的代码不会再执行
    }
}

2. 发起重定向的 Servlet (RedirectServlet.java)

复制代码
// RedirectServlet.java
@WebServlet("/redirectServlet")
publicclassRedirectServletextendsHttpServlet {
    protectedvoiddoGet(HttpServletRequest request, HttpServletResponse response)throws ServletException, IOException {
        System.out.println("RedirectServlet: 接待了请求,正在处理...");

        // redirect 无法通过 request 域传递数据,因为它是两次不同的请求
        // 下面这行代码设置的属性将会丢失
        request.setAttribute("message", "这个秘密文件将会丢失!");

        System.out.println("RedirectServlet: 告诉浏览器 '请您去另一个地址' (FinalDestinationServlet)。");

        // 执行 redirect
        response.sendRedirect(request.getContextPath() + "/finalDestination");
    }
}

3. 最终目的地 Servlet (FinalDestinationServlet.java)

复制代码
// FinalDestinationServlet.java
@WebServlet("/finalDestination")
publicclassFinalDestinationServletextendsHttpServlet {
    protectedvoiddoGet(HttpServletRequest request, HttpServletResponse response)throws ServletException, IOException {
        System.out.println("FinalDestinationServlet: 我是最终目的地!");

        // 尝试从 request 域中获取数据
        Stringmessage= (String) request.getAttribute("message");

        response.setContentType("text/html;charset=UTF-8");
        PrintWriterout= response.getWriter();
        out.println("<h1>欢迎来到最终目的地!</h1>");
        
        if (message != null) {
            out.println("<p>我收到了秘密文件: " + message + "</p>");
            out.println("<p><b>说明:</b> 这是通过 forward 过来的,因为 request 对象是同一个。</p>");
        } else {
            out.println("<p>我没有收到任何秘密文件。</p>");
            out.println("<p><b>说明:</b> 这是通过 redirect 过来的,浏览器发起了全新的请求,旧 request 的数据丢失了。</p>");
        }
    }
}
相关推荐
独立开阀者_FwtCoder1 小时前
前端大一统时代来了
前端·vue.js·github
测试开发技术2 小时前
git rm 命令与系统的 rm 命令有什么区别?
git·gitlab·github·面试题
longze_73 小时前
Jenkins credentials 增加了github credential 但是在Git SCM 凭证中不显示
git·github·jenkins
seth3 小时前
coding 要停服了, 把所有 CI 迁移到 Github 上 (服务器无需翻墙)
ci/cd·docker·github
前端拿破轮4 小时前
重生之我在掘金做毕设【一】:技术选型
前端·github·前端工程化
ai小鬼头17 小时前
如何重装旁路由系统并优化AIStarter部署:一步步教程
java·css·github
DogDaoDao18 小时前
2025年 GitHub 主流开源视频生成模型介绍
人工智能·深度学习·开源·大模型·github·音视频·视频生成
qianmoQ19 小时前
GitHub 趋势日报 (2025年07月13日)
github
天氰色等烟雨20 小时前
支持MCP服务的多平台一键发布工具
大数据·github·mcp