${pageContext.request.contextPath}和<%=request.getContextPath()%>

本文介绍在JSP中使用${pageContext.request.contextPath}

${pageContext.request.contextPath}和<%=request.getContextPath()%>


       做的一个web项目,需要在jsp页面中获取js、css和图片等资源,本想采用相对路径来写,但是发现可移植性太差,在自己电脑上还好使,但辛辛苦苦调好代码后,放到其他电脑上又得再次辛辛苦苦修改相关路径。于是决定采用绝对路径来写。然后在网上找寻相关方法,发现用的比较多的两个:${pageContext.request.contextPath}和<%=request.getContextPath()%>,但具体用哪个我也不大清楚,于是继续查找两者的区别,但让我郁闷的是,网上“抄袭”的真多啊!而且说了一大堆!尽是些不痛不痒的专业名词,关键的真没几个!所以我决定靠自己理解,现在明白了!我想用一种比较通俗的语言分享一下我对此的认识!

      可以说${pageContext.request.contextPath}等价于<%=request.getContextPath()%>!它们的意思就是取得当前项目名称(或者是--取出部署的应用程序名,不过这么说太官方了,试问有多少人知道“取出部署的应用程序名”的义)
      那么什么是当前的项目名称呢?

      

      假定你的web应用名称为hotel,这个hotel就是当前的项目名称,不过你在浏览器中输入请求路径时,例如输入http//:localhost:8080/hotel/login.jsp 

      ${pageContext.request.contextPath}或<%=request.getContextPath()%>就是从这个请求路径(URL)上截取(是截取) /hotel ,看清楚,前面是有"/",而这个“/”代表的是"http//:localhost:8080",看清楚这里是没有"/"的!

      对应到真是的物理地址,即为图中WebContent目录!

      另外,如果你是在Tomcat的server.xml文件中配置了虚拟目录,例如

      

      那么我们在对应的物理目录F:\javaweb中创建test_jsp.jsp文件,内容为

       

      开启Tomcat,进行测试,发现输出结果为

      

      可以看到,此时输出为虚拟目录,并且二者完全等价!

      因此,在表单<formaction="${pageContext.request.contextPath}/hotel/login.jsp">中,这样写路径永远对,翻译过来${pageContext.request.contextPath}/hotel/login.jsp其中的含义,就是http//:localhost:8080/hotel/login.jsp,相当于你写了一全路径!当然前提是你的JSP页面等等必须放置的位置正确才可以,所以才说明路径永远正确。

       为什么这么要做呢?因为学过一些MVC模式的程序员都知道,JSP的作用是用来显示的(表现的),其他的作用最好不要在这个页面上显示,尤其是Java代码!所以就用EL表达式来替代类似有“<%%>”这样的代码格式。


&lt;%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%&gt; &lt;%@ page import="javax.servlet.*,java.text.*" %&gt; &lt;% String path = request.getContextPath(); String basePath = request.getScheme()+"://"+request.getServerName()+":"+request.getServerPort()+path+"/"; %&gt; &lt;!DOCTYPE html> &lt;html> &lt;head> &lt;meta charset="utf-8"> &lt;meta name="viewport" content="width=device-width, initial-scale=1.0"> &lt;meta name="renderer" content="webkit"> &lt;title>收入综合分析平台&lt;/title> &lt;link rel="stylesheet" href="${pageContext.request.contextPath}/resources/bootstrap-3.3.7-dist/css/bootstrap.min.css"> &lt;%-- 右击菜单样式 --%&gt; &lt;link href="${pageContext.request.contextPath}/resources/ruoyi/ruoyi/css/jquery.contextMenu.min.css" rel="stylesheet"/> &lt;%-- &lt;link href="${pageContext.request.contextPath}/resources/ruoyi/ruoyi/css/animate.min.css" rel="stylesheet"/>--%&gt; &lt;link href="${pageContext.request.contextPath}/resources/ruoyi/ruoyi/css/style.min.css" rel="stylesheet"/> &lt;link href="${pageContext.request.contextPath}/resources/ruoyi/ruoyi/css/skins.css" rel="stylesheet"/> &lt;link href="${pageContext.request.contextPath}/resources/ruoyi/ruoyi/css/ry-ui.css?v=4.6.1" rel="stylesheet"/> &lt;%-- 图标库 --%&gt; &lt;link href="${pageContext.request.contextPath}/resources/ruoyi/ruoyi/css/font-awesome.min.css" rel="stylesheet"/> &lt;script src="${pageContext.request.contextPath}/resources/js/jquery-1.11.3.min.js?v=&lt;%=System.currentTimeMillis()%&gt;">&lt;/script> &lt;script src="${pageContext.request.contextPath}/resources/bootstrap-3.3.7-dist/js/bootstrap.min.js?v=&lt;%=System.currentTimeMillis()%&gt;">&lt;/script> &lt;script src="${pageContext.request.contextPath}/resources/global/global.js?v=&lt;%=System.currentTimeMillis()%&gt;">&lt;/script> &lt;script type="text/javascript"> $(function(){ $.ajax({ type: 'get', url: "dic/getUrlByUser", async: false, dataType: 'json', success: function (data) {//返回list数据并循环获取 var ht = ""; var arr = ['desktop','calculator','leanpub','map','send','bar-chart','line-chart','train','american-sign-language-interpreting','gears','user-circle-o','paper-plane','firefox']; for(var i=0;i&lt;data.length;i++){ //循环一级菜单 ht = ht +'&lt;li>'+ '&lt;a href="#">' + '&lt;i class="fa fa-'+arr[i]+'">&lt;/i> ' + '&lt;span class="nav-label">'+data[i].mkName+'&lt;/span>' + '&lt;span class="fa arrow">&lt;/span>' + '&lt;/a>'+ '&lt;ul class="nav nav-second-level collapse">'; var obj = data[i].obj; if(obj[0].cdlb == ""){//该模块不存在二级菜单 for(var j=0;j&lt;obj.length;j++){ ht = ht + '&lt;li>\n' + '&lt;a class="menuItem" href="'+obj[j].cdUrl+'">'+ obj[j].cdName +'&lt;/a>' + '&lt;/li>'; } ht = ht + '&lt;/ul>&lt;/li>'; }else{ var cdlbArr = new Array(); for(var j=0;j&lt;obj.length;j++){ cdlbArr.push(obj[j].cdlb); } cdlbArr = uniqueArr(cdlbArr); for(var k=0;k&lt;cdlbArr.length;k++){ ht = ht + '&lt;li>\n' + '&lt;a href="#">&lt;i class="fa fa-pied-piper">&lt;/i> '+cdlbArr[k]+'&lt;span class="fa arrow">&lt;/span>&lt;/a>\n' + '&lt;ul class="nav nav-third-level collapse">'; for(var m=0;m&lt;obj.length;m++){ if(cdlbArr[k] == obj[m].cdlb){ ht = ht + '&lt;li>\n' + '&lt;a class="menuItem" href="'+obj[m].cdUrl+'">'+ obj[m].cdName +'&lt;/a>' + '&lt;/li>'; } } ht = ht + '&lt;/ul>&lt;/li>'; } } ht = ht + '&lt;/ul>&lt;/li>'; } $("#side-menu").append(ht); } }); }) /* * 用于解决浏览器关闭后Cookie未失效,攻击者可在用户关闭浏览器后,通过同一cookie直接访问网站(无需重新登录),窃取用户会话; * 由于ajax是异步 使用时会造成发送不过去的现象 需要使用同步请求 * 另外 如果只是使用 window.addEventListener('beforeunload'来判断 会出现 刷野也会消除session 所以只能通过一下方法来实现 * */ window.addEventListener('beforeunload', function(e) { // 获取导航类型(替代废弃的performance.navigation.type) console.log(JSON.stringify(performance.getEntriesByType("navigation")[0].toJSON().type)) const navType = performance.getEntriesByType("navigation")[0].toJSON().type; sendLogoutRequest(navType) // 排除刷新行为,仅在离开/关闭时触发逻辑 if (navType !== 'reload') { console.log('离开页面,执行逻辑') //sendLogoutRequest() return '111'; } }); /*class PageLeaveHandler { constructor() { this.isRefreshing = false; this.pendingUnload = false; this.pendingUnloadnum = 0;//默认值 this.init(); } init() { // 监听刷新快捷键 window.addEventListener('keydown', (e) => { if (e.key === 'F5' || (e.ctrlKey && e.key === 'F5')) { this.isRefreshing = true; } }); // 监听beforeunload事件 window.addEventListener('beforeunload', this.handleBeforeUnload.bind(this)); } handleBeforeUnload(e) { if (this.isRefreshing) { this.pendingUnload = false; return; } //sendLogoutRequest(this.isRefreshing) this.pendingUnload = true; return 1; } } new PageLeaveHandler();*/ function sendLogoutRequest(navType) { const logoutUrl = '${pageContext.request.contextPath}/sys/loginoutall?navType='+navType; if (navigator.sendBeacon) { navigator.sendBeacon(logoutUrl, 'loginoutall'); } else { const xhr = new XMLHttpRequest(); xhr.open('GET', logoutUrl, false); // 同步请求 xhr.timeout = 2000; try { xhr.send(); } catch (e) { // 忽略错误 } } } /*window.addEventListener('beforeunload', function(e) { // 取消事件的默认行为(部分浏览器需要) e.preventDefault(); if(!e.isTrusted){ // 2. 拼接正确的上下文路径(关键:避免404) const logoutUrl = '${pageContext.request.contextPath}/sys/loginoutall'; // 3. 用sendBeacon发送请求,浏览器会优先保证发送完成 if (navigator.sendBeacon) { navigator.sendBeacon(logoutUrl, 'loginoutall'); } else { const xhr = new XMLHttpRequest(); xhr.open('GET', logoutUrl, false); // 同步请求 xhr.timeout = 2000; try { xhr.send(); } catch (e) { } } } });*/ &lt;/script> &lt;/head> &lt;body class="fixed-sidebar full-height-layout gray-bg skin-blue theme-light" style="overflow: hidden"> &lt;div id="wrapper"> &lt;!--左侧导航开始--> &lt;nav class="navbar-default navbar-static-side" role="navigation"> &lt;div class="nav-close"> &lt;i class="fa fa-times-circle">&lt;/i> &lt;/div> &lt;a href=""> &lt;li class="logo hidden-xs"> &lt;span class="logo-lg">欢迎${user.username}&lt;/span> &lt;/li> &lt;/a> &lt;div class="sidebar-collapse"> &lt;ul class="nav" id="side-menu"> &lt;li> &lt;div class="user-panel"> &lt;%-- 该位置换谁个人中心地址 --%&gt; &lt;a class="menuItem noactive" title="个人中心" href=""> &lt;div class="hide" text="个人中心">&lt;/div> &lt;div class="pull-left image"> &lt;%--&lt;img src="resources/ruoyi/favicon.ico" class="img-circle" alt="User Image">--%&gt; &lt;img src="resources/image/eee.png" class="img-circle" alt="User Image"> &lt;/div> &lt;/a> &lt;div class="pull-left info"> &lt;%-- 该位置可以加上el表达式 然后读取成登录名即可 --%&gt; &lt;p>&lt;/p> &lt;a href="#">&lt;i class="fa fa-circle text-success">&lt;/i> 在线&lt;/a> &lt;%-- 该位置发起退出请求 换上即可 --%&gt; &lt;a href="sys/loginout" style="padding-left:5px;">&lt;i class="fa fa-sign-out text-danger">&lt;/i> 注销&lt;/a> &lt;/div> &lt;/div> &lt;/li> &lt;li> &lt;a class="menuItem" href="web/main/sy.jsp">&lt;i class="fa fa-home">&lt;/i> &lt;span class="nav-label">首页&lt;/span>&lt;/a> &lt;/li> &lt;/ul> &lt;/div> &lt;/nav> &lt;!--左侧导航结束--> &lt;!--右侧部分开始--> &lt;div id="page-wrapper" class="gray-bg dashbard-1"> &lt;div class="row border-bottom"> &lt;nav class="navbar navbar-static-top" role="navigation" style="margin-bottom: 0"> &lt;div class="navbar-header"> &lt;a class="navbar-minimalize minimalize-styl-2" style="color:#FFF;" href="#" title="收起菜单"> &lt;i class="fa fa-bars">&lt;/i> &lt;/a> &lt;/div> &lt;ul class="nav navbar-top-links navbar-right welcome-message"> &lt;%--&lt;li>&lt;a data-toggle="tooltip" data-trigger="hover" data-placement="bottom" title="开发文档" href="http://doc.ruoyi.vip/ruoyi" target="_blank">&lt;i class="fa fa-question-circle">&lt;/i> 文档&lt;/a>&lt;/li> &lt;li>&lt;a data-toggle="tooltip" data-trigger="hover" data-placement="bottom" title="锁定屏幕" href="#" id="lockScreen">&lt;i class="fa fa-lock">&lt;/i> 锁屏&lt;/a>&lt;/li>--%&gt; &lt;li>&lt;a data-toggle="tooltip" data-trigger="hover" data-placement="bottom" title="全屏显示" href="#" id="fullScreen">&lt;i class="fa fa-arrows-alt">&lt;/i> 全屏&lt;/a>&lt;/li> &lt;%--&lt;li class="dropdown user-menu"> &lt;a href="javascript:void(0)" class="dropdown-toggle" data-hover="dropdown"> &lt;img src="resources/image/eee.png" class="user-image"> &lt;%– 该位置用el表达式读取登录名 –%&gt; &lt;span class="hidden-xs">海来怡天&lt;/span> &lt;/a> &lt;ul class="dropdown-menu"> &lt;li class="mt5"> &lt;%– 该位置是个人中心超链接 –%&gt; &lt;a href="" class="menuItem noactive"> &lt;i class="fa fa-user">&lt;/i> 个人中心&lt;/a> &lt;/li> &lt;li> &lt;a onclick="resetPwd()"> &lt;i class="fa fa-key">&lt;/i> 修改密码&lt;/a> &lt;/li> &lt;li> &lt;a onclick="switchSkin()"> &lt;%&lt;a id="btn" >–%&gt; &lt;i class="fa fa-dashboard">&lt;/i> 切换主题 &lt;/a> &lt;/li> &lt;li class="divider">&lt;/li> &lt;li> &lt;%– 该位置是推出登录请求 –%&gt; &lt;a href=""> &lt;i class="fa fa-sign-out">&lt;/i> 退出登录&lt;/a> &lt;/li> &lt;/ul> &lt;/li>--%&gt; &lt;/ul> &lt;/nav> &lt;/div> &lt;div class="row content-tabs"> &lt;button class="roll-nav roll-left tabLeft"> &lt;i class="fa fa-backward">&lt;/i> &lt;/button> &lt;nav class="page-tabs menuTabs"> &lt;div class="page-tabs-content"> &lt;a href="javascript:" class="active menuTab" data-id="web/main/sy.jsp">首页&lt;/a> &lt;/div> &lt;/nav> &lt;button class="roll-nav roll-right tabRight"> &lt;i class="fa fa-forward">&lt;/i> &lt;/button> &lt;a href="javascript:void(0);" class="roll-nav roll-right tabReload">&lt;i class="fa fa-refresh">&lt;/i> 刷新&lt;/a> &lt;/div> &lt;a id="ax_close_max" class="ax_close_max" href="#" title="关闭全屏"> &lt;i class="fa fa-times-circle-o">&lt;/i> &lt;/a> &lt;div class="row mainContent" id="content-main" > &lt;iframe class="RuoYi_iframe" name="iframe0" width="100%" height="100%" data-id="web/main/sy.jsp" src="web/main/sy.jsp" frameborder="0" seamless>&lt;/iframe> &lt;/div> &lt;/div> &lt;!--右侧部分结束--> &lt;/div> &lt;!-- 全局js --> &lt;script type="text/javascript" src="${pageContext.request.contextPath}/resources/ruoyi/js/plugins/metisMenu/jquery.metisMenu.js?v=&lt;%=System.currentTimeMillis()%&gt;">&lt;/script> &lt;script type="text/javascript" src="${pageContext.request.contextPath}/resources/ruoyi/js/plugins/slimscroll/jquery.slimscroll.min.js?v=&lt;%=System.currentTimeMillis()%&gt;">&lt;/script> &lt;script type="text/javascript" src="${pageContext.request.contextPath}/resources/ruoyi/js/jquery.contextMenu.min.js?v=&lt;%=System.currentTimeMillis()%&gt;">&lt;/script> &lt;%--&lt;script src="${pageContext.request.contextPath}/resources/ruoyi/ajax/libs/layui/layui.js?v=&lt;%=System.currentTimeMillis()%&gt;">&lt;/script>--%&gt; &lt;%-- 该方法用于将页面加载到本页面 --%&gt; &lt;script type="text/javascript" src="${pageContext.request.contextPath}/resources/ruoyi/ajax/libs/blockUI/jquery.blockUI.js?v=&lt;%=System.currentTimeMillis()%&gt;">&lt;/script> &lt;script type="text/javascript" src="${pageContext.request.contextPath}/resources/ruoyi/ajax/libs/layer/layer.min.js?v=&lt;%=System.currentTimeMillis()%&gt;">&lt;/script> &lt;script type="text/javascript" src="${pageContext.request.contextPath}/resources/ruoyi/ruoyi/js/ry-ui.js?v=&lt;%=System.currentTimeMillis()%&gt;">&lt;/script> &lt;script type="text/javascript" src="${pageContext.request.contextPath}/resources/ruoyi/ruoyi/js/common.js?v=&lt;%=System.currentTimeMillis()%&gt;">&lt;/script> &lt;script type="text/javascript" src="${pageContext.request.contextPath}/resources/ruoyi/ruoyi/index.js?v=&lt;%=System.currentTimeMillis()%&gt;">&lt;/script> &lt;script type="text/javascript" src="${pageContext.request.contextPath}/resources/ruoyi/ajax/libs/fullscreen/jquery.fullscreen.js?v=&lt;%=System.currentTimeMillis()%&gt;">&lt;/script> &lt;script> /*window.history.forward(1); var ctx = "/zhfx"; // 皮肤缓存 var skin = storage.get("skin");*/ // history(表示去掉地址的#)否则地址以"#"形式展示 var mode = "history"; // 历史访问路径缓存 var historyPath = storage.get("historyPath"); // 是否页签与菜单联动 var isLinkage = true; /* * 这个js如果删了 按下F12编辑时 页面会不起作用 * */ $(function() { var lockPath = storage.get('lockPath'); if($.common.equals("history", mode) && window.performance.navigation.type == 1) { var url = storage.get('publicPath'); if ($.common.isNotEmpty(url)) { applyPath(url); } } else if($.common.isNotEmpty(lockPath)) { applyPath(lockPath); storage.remove('lockPath'); } else { var hash = location.hash; if ($.common.isNotEmpty(hash)) { var url = hash.substring(1, hash.length); applyPath(url); } else { if($.common.equals("history", mode)) { storage.set('publicPath', ""); } } } $("[data-toggle='tooltip']").tooltip(); }); &lt;/script> &lt;/body> &lt;/html> 为什么 一个刷新 一个跳转 这俩 点刷新是reload 再点跳转还是reload 必须第二下才是navigate 但是这样点了navigate之后 再点刷新还是navigate 必须再点刷新才是reload 这可不能这样
最新发布
11-12
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值