getRequestDispatcher(path).forward(),,执行完,后面的代码居然还会执行!!!记得加return 啊亲

本文详细对比了HttpServletResponse.sendRedirect方法实现的请求重定向与RequestDispatcher.forward方法实现的请求转发。从适用范围、浏览器地址栏变化、请求过程比喻、request对象使用及代码执行流程等方面进行了阐述。

尽管HttpServletResponse.sendRedirect方法和RequestDispatcher.forward方法都可以让浏览器获得另外一个URL所指向的资源,但两者的内部运行机制有着很大的区别。下面是HttpServletResponse.sendRedirect方法实现的请求重定向与RequestDispatcher.forward方法实现的请求转发的总结比较:

(1)RequestDispatcher.forward方法只能将请求转发给同一个WEB应用中的组件;而 HttpServletResponse.sendRedirect 方法不仅可以重定向到当前应用程序中的其他资源,还可以重定向到同一个站点上的其他应用程序中的资源,甚至是使用绝对URL重定向到其他站点的资源。如果传递给HttpServletResponse.sendRedirect 方法的相对URL以“/”开头,它是相对于整个WEB站点的根目录 ;如果创建RequestDispatcher对象时指定的相对URL以“/”开头,它是相对于当前WEB应用程序的根目录

(2)调用HttpServletResponse.sendRedirect方法重定向的访问过程结束后,浏览器地址栏中显示的URL会发生改变,由初始的URL地址变成重定向的目标URL;而调用RequestDispatcher.forward 方法的请求转发过程结束后,浏览器地址栏保持初始的URL地址不变。

(3)HttpServletResponse.sendRedirect方法对浏览器的请求直接作出响应,响应的结果就是告诉浏览器去重新发出对另外一个URL的访问请求,这个过程好比有个绰号叫“浏览器”的人写信找张三借钱,张三回信说没有钱,让“浏览器”去找李四借,并将李四现在的通信地址告诉给了“浏览器 ”。于是,“浏览器”又按张三提供通信地址给李四写信借钱,李四收到信后就把钱汇给了“浏览器”。可见,“浏览器”一共发出了两封信和收到了两次回复,“ 浏览器”也知道他借到的钱出自李四之手。RequestDispatcher.forward方法在服务器端内部将请求转发给另外一个资源,浏览器只知道发出了请求并得到了响应结果,并不知道在服务器程序内部发生了转发行为。这个过程好比绰号叫“浏览器”的人写信找张三借钱,张三没有钱,于是张三找李四借了一些钱,甚至还可以加上自己的一些钱,然后再将这些钱汇给了“浏览器”。可见,“浏览器”只发出了一封信和收到了一次回复,他只知道从张三那里借到了钱,并不知道有一部分钱出自李四之手。

(4)RequestDispatcher.forward方法的调用者与被调用者之间共享相同的request对象和response对象,它们属于同一个访问请求和响应过程 ;而HttpServletResponse.sendRedirect方法调用者与被调用者使用各自的request对象和response对象,它们属于两个独立的访问请求和响应过程 。对于同一个WEB应用程序的内部资源之间的跳转,特别是跳转之前要对请求进行一些前期预处理,并要使用HttpServletRequest.setAttribute方法传递预处理结果,那就应该使用 RequestDispatcher.forward方法。不同WEB应用程序之间的重定向,特别是要重定向到另外一个WEB站点上的资源的情况,都应该使用HttpServletResponse.sendRedirect方法。

(5)无论是RequestDispatcher.forward方法,还是HttpServletResponse.sendRedirect方法,在调用它们之前,都不能有内容已经被实际输出到了客户端。如果缓冲区中已经有了一些内容,这些内容将被从缓冲区中清除。

 

(6) 代码的执行:

无论是 request.getRequestDispatcher(path).forward(request, response)还是response.sendRedirect,程序都会在执行完该句的情况下继续向下执行,因此在必要的时候应该使用return终止该方法.

对于 request.getRequestDispatcher(path).forward(request, response),在执行完该方法的时候再进行对request的操作已经没有任何意义,如果在该方法之后再进行request.setAttribute(),该值将不会被放进当前请求的request中.

response.setRedirect:该方法执行之后,接下来的方法也会被执行.但是使用该方法的时候,会发送一个全新的request,将不再使用原先的request,因此不论在该方法执行之前,还是在该方法执行之后,对request操作,都是无效的.

private static final String PROJECT_SEL_JSP_PATH = "/page/menu.jsp"; protected void login(HttpServletRequest request, HttpServletResponse response) throws jakarta.servlet.ServletException, java.io.IOException { log.debug("ログイン処理 start"); request.removeAttribute(LOGIN_ERR_ATTR); try { String houzinCode = StringFunc.getUnicodeString(request.getParameter("HCD")); String userId = StringFunc.getUnicodeString(request.getParameter("user")); String password = StringFunc.getUnicodeString(request.getParameter("password")); // ログイン画面よりログイン時 Map<String, List<String>> resultMsg = new HashMap<>(); List<Object> result = LoginCheck.loginCheck(userId, password); DateTimeFormatter sdf = DateTimeFormatter.ofPattern(ConstInfo.USER_PASSWORD.DATA_FORMATTER); int size = 0; // ストアドプロシージャを決定 if (result != null && !result.isEmpty()) { size = result.size(); // 長さは正常に返されました if (size == 2) { Object secondElement = result.get(1); if (secondElement instanceof String) { sdf.parse((String) secondElement); return; } } else { // エラーコードのトラバース for (Object element : result) { if (element instanceof String) { String strresult = (String) element; if (strresult != null && !strresult.isEmpty()) { switch (strresult) { case Constants.USER_ID_REQUIRED_CODE: errTextMsgSet(resultMsg, HomeServiceStatus.USERID_ERR, Constants.USER_ID_REQUIRED); break; case Constants.PASSWORD_REQUIRED_CODE: errTextMsgSet(resultMsg, HomeServiceStatus.PASSWORD_ERR, Constants.PASSWORD_REQUIRED); break; case Constants.INVALID_CREDENTIALS_CODE: case Constants.INVALID_CREDENTIALS_CODE_6: errTextMsgSet(resultMsg, HomeServiceStatus.USERID_ERR, Constants.INVALID_CREDENTIALS); break; case Constants.ACCOUNT_LOCKED_CODE: errTextMsgSet(resultMsg, HomeServiceStatus.USERID_ERR, Constants.ACCOUNT_LOCKED); break; case Constants.MAX_LOGIN_ATTEMPTS_CODE: errTextMsgSet(resultMsg, HomeServiceStatus.USERID_ERR, Constants.MAX_LOGIN_ATTEMPTS); break; case Constants.USER_DISABLED_CODE: errTextMsgSet(resultMsg, HomeServiceStatus.USERID_ERR, Constants.USER_DISABLED); break; case Constants.PASSWORD_EXPIRED_CODE: errTextMsgSet(resultMsg, HomeServiceStatus.USERID_ERR, Constants.PASSWORD_EXPIRED); break; case Constants.SYSTEM_ERROR_CODE: errTextMsgSet(resultMsg, HomeServiceStatus.USERID_ERR, Constants.SYSTEM_ERROR); break; } } } } } } // resultMsg空です if (!resultMsg.isEmpty()) { Gson gson = new Gson(); String jsonStr = gson.toJson(resultMsg); request.setAttribute("user", userId); request.setAttribute("error", jsonStr); request.getRequestDispatcher(LOGIN_JSP_PATH).forward(request, response); } else { request.setAttribute("LOGIN_CORP", houzinCode); request.setAttribute("LOGIN_USER", userId); Date lastLoginTime = (Date) result.get(1); HttpSession ses = request.getSession(true); ses.setAttribute("sysLastLoginDateTime", lastLoginTime); ses.setAttribute("APP_USER", userId); AuthUser bi = new AuthUser(); bi.userId = userId; bi.Address = "::1"; bi.yukoDateCnt = 2; if (!CMM010DBDA.NSH_INF_INS(bi)) { BizException.throwException(BizExceptionMsg.AUTH_ERROR); } List<String> newShainCdList = getNewShainCd(userId); if (!newShainCdList.isEmpty()) { CommonMethod.cfbGetControlInfo(ses); CommonMethod.cfbGetUserInfo(newShainCdList.get(1), ses); } servletContext.getRequestDispatcher(PROJECT_SEL_JSP_PATH).forward(request, response); } } catch (Exception e) { BizException.throwException(BizExceptionMsg.OTHER_ERR); } log.debug("ログイン処理 end"); } 在上面代码中login登录后,执行【servletContext.getRequestDispatcher(PROJECT_SEL_JSP_PATH).forward(request, response);】后直接跳转到menu.jsp页面了, 如何修改,让他跳转到刚刚你说的MenuServlet?
10-18
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值