|
一、使用绝对与相对路径的讨论、路径题目犯错来源、路径题目解决方案
JSP 中毕竟采用绝对路径仍是采用相对路径跟着所采用技术的越来越复杂,这个题目也变得越来越难以解决。
1、采用相对路径碰到的题目
1) 相对路径虽然比较灵活,但假如想复制页面内的代码却变得比较难题,由于不同的页面具有不同的相对路径,
复制后必需修改每一个连接的路径。
2)假如页面被多于一个的页面所包含,那么被包含页面中的相对路径将是不准确的。
假如采用 Struts 的 Action 返回页面,那么因为页面路径与 Action 路径不同,使得浏览器无法准确解释页面中的路径,
如页面为/pages/cust/cust.jsp ,图片所有目录为 /images/title.gif ,
这时在 /pages/cust/cust.jsp 中的所用的路径为 ”../../images/title.gif” ,
但是假如某一个 Action 的Forward 指向这个 JSP 文件,而这个 Action 的路径为/cust/manage.do ,
那么页面内容中 ”../../images/title.gif” 就不再指向准确的路径了。
解决以上题目好像只有使用绝对路径了。
2、采用绝对路径碰到的题目
1) 跟着不同的 Web 应用发布方式,绝对路径的值也不同。
如Web 应用发布为 MyApp ,则路径 ”/MyApp/images/title.gif” 是准确的,
但发布为另一应用时如 MyApp2 ,这个路径就分歧错误了,也许这个情况比较少,
但以default 方式发布 Web 应用时以上绝对路径也不同: ”/images/title.gif” 。 (就是域名默认的访问的web应用程序)
3.解决方案
1)采用绝对路径,但为了解决不同部署方式的差别,
在所有非struts 标签的路径前加 ${pageContext.request.contextPath} ,如原路径为:
”/images/title.gif” ,改为“${pageContext.request.contextPath}/images/title.gif” 。
代码 ”${pageContext.request.contextPath}” 的作用是掏出部署的应用程序名,这样无论如何部署,所用路径都是准确的。
缺点:操纵不便,其他工具无法准确解释 ${pageContext.request.contextPath}
2)采用相对路径,在每个JSP 文件中加入 base 标签,如:
<basehref="http://${header['host']}${pageContext.request.contextPath}/pages/cust/relation.jsp"/>
这样所有的路径都可以使用相对路径。
缺点:对于被包含的文件依然无效。
真正使用时需要灵活应用1)和2),写出更加健壮的代码。
二、在jsp中常常会碰到路径的题目,泛起404的错误或者是图片无法显示.
这个题目可以从以下几个方面来解决:
1 、链接或表单提交的路径
2、重定向和转发的路径
3、在MyEclipse有jsp的两种模板advanced 和base Templates的路径题目
以下为解决之道
1、表单提交和链接都是从客户端哀求而来
/指的Tomcat的根目录,写绝对路径应该写成"/当前Web程序根名称/资源名" 如"/WebModule1/jsp1.jsp","/bbs/Servlet/MyServlet"。
2、重定向和转发的路径题目
1)重定向response.sendRedirect("")是服务器向客户端发送一个哀求头信息,由客户端再哀求一次服务器.因此情况同上述1的一样,(假如是相对路径会转变成绝对路径)
2)转发是在服务器内部进行的,写绝对路径/开头指的是当前的Web应用程序.绝对路径写法就是"/jsp1.jsp"或"/servlet/MyServlet"
总结:以上两点要留意是区分是从服务器外的哀求,还在是内部转发,
从服务器外的哀求,从Tomcat根写起(就是要包括当前Web的根),
是服务器内部的转发,很简朴了,由于在当前服务器内,/写起指的就是当前Web的根.
3、在MyEclispe中的JSP的两种模板,一种是Advanced,一种是Basic
两者的区别也会涉及到路径题目,如图片的路径,链接的路径
Advanced Template中页面的代码多了以下代码:
<%
String path = request.getContextPath();
String basePath = request.getScheme()+"://"+request.getServerName()+":"+request.getServerPort()+path+"/";
%>
<base href="<%=basePath%>" _cke_saved_href="<%=basePath%>">
其中<base href="<%=basePath%>">必需放在<head></head>中
这段代码的作用是设置基础路径的,basepath为变量.<base href="">标签的用处是解决编程时候的相对路径题目.
设置了<base href="<%=basePath%>">
页面上图片,链接的路径指的都是当前应用程序的根路径下.
当然,这个base还有一个用法,如在head部门加上这么一行:<base href="_blank">,就是默认所有链接在新窗口打开。
按以上方法可解决所有的路径题目.
三、示例
JSP页面中的相对链接
在JSP中的使用 "相对路径" 则有可能会泛起题目.由于 网页中的 "相对路径" , 他是相对于 "URL哀求的地址" 去寻找资源.
举个例子:
如果我们有一个项目: MyApp,在该项目下, 有一个jsp文件夹
该文件夹下包括:
login.jsp // 登陆页面
register.jps // 注册页面
我们在浏览器中输入地址 (留意: 地址的内容):
http://localhost:8080/MyApp/jsp/login.jsp
这时候, 浏览器会链接到 "登陆页面" (login.jsp)
在login.jsp文件内包含了如下 "部门代码":
<a href="jsp/register.jsp">注册用户</a>
那么,假如我们点击这个链接,就会在浏览器地址栏中, 泛起如下错误链接:
http://localhost:8080/MyApp/jsp/jsp/register.jsp
为什么会泛起"/jsp/jsp/register.jsp"呢?
由于, 网页中的"相对链接", 是相对于你所 "哀求的URL路径" 所决定的.
即:由于这里哀求路径是:http://localhost:8080/MyApp/jsp/login.jsp
那么, 浏览器 就会在这个路径下(即:http://localhost:8080/MyApp/jsp/)去找 jsp/register.jsp
所以就会泛起如下错误内容:
http://localhost:8080/MyApp/jsp/jsp/register.jsp
上面的题目,就是调用页面和被调用页面的URL不同所造成的,
此类错误也经常会泛起在2个页面之间进行 "转发"(forward) 操纵的时候。
由于forward是在后台进行的,对客户端来说是透明的。(即: URL不改变,而数据内容却是另一个页面返归来的。。。)
方法一:直接采用绝对路径 (不推荐)
在JSP页面端,获得本项目的绝对地址(假如你的项目叫MyApp,那么获得到的地址就是 http://localhost:8080/MyApp/):
代码如下:
<%@ page language="java" pageEncoding="GBK" contentType="text/HTML;charset=gbk" isELIgnored="false"%>
<%
String path = request.getContextPath();
// 获得本项目的地址(例如: http://localhost:8080/MyApp/)赋值给basePath变量
String basePath = request.getScheme()+"://"+request.getServerName()+":"+request.getServerPort()+path+"/";
// 将 "项目路径basePath" 放入pageContext中,待以后用EL表达式读出。
pageContext.setAttribute("basePath",basePath);
%>
<html>
<head></head>
<body>
<a href="${pageScope.basePath}jsp/register.jsp">
</body>
</html>
我们可以看到,在标签<a>中的href属性内,我们直接采用了
“本项目路径 ${pageScope.basePath}” 加上 "jsp/register.jsp" ,
从而构成一个绝对路径(即: http://localhost:8080/MyApp/jsp/register.jsp)
但是这样做有一个很不好的地方,那就是我们必需要在每个链接的前面都要加上 “${pageScope.basePath}”
假如这样做的话,将是一件很可怕的事情。
方法二: 利用html中的<base>标签(推荐)
下面是对html中的<base>的先容:
base 元素可划定页面中所有链接的基准 URL
默认情况下,页面中的链接(包括样式表、脚本和图像的地址)都是相对于当前页面的地址(即:浏览器地址栏里的哀求URL)。
我们可以使用<base>标签中的href属性来设置,所有的“相对基准 URL”。
这是JSP真个代码,但是这里我们并没有采用 ${pageScope.basePath}+"相对路径地址" 的方法,
而是采用了html文件中的<base>标签:
代码如下:
<%@ page language="java" pageEncoding="GBK" contentType="text/html;charset=gbk" isELIgnored="false"%>
<%
String path = request.getContextPath();
// 获得项目完全路径(假设你的项目叫MyApp,那么获得到的地址就是 http://localhost:8080/MyApp/):
String basePath = request.getScheme()+"://"+request.getServerName()+":"+request.getServerPort()+path+"/";
%>
<html>
<head>
<!-- base需要放到head中 -->
<base href="<%=basePath%>">
</head>
// 这里我们就可以直接使用相对路径(即: 相对于base标签)
<a href="jsp/login.jsp">Login</a>
</html>
当我们去执行上面的那段JSP代码后,我们可以在浏览器中可以查看,他所返回给客户真个html代码:
执行完上述JSP后,所返回的html代码如下:
<html>
<head>
<base href="http://localhost:8080/MyApp/">
</head>
// 设置了<base>后,相对路径,相对于的就是base中的路径,而不再是浏览器地址的哀求路径啦~~~
<a href="jsp/login.jsp">Login</a>
</html>
我们可以看到JSP返回的html代码中,包含了<base href="http://localhost:8080/MyApp/">内容。
也就是说,在本html文件中,碰到的所有 “相对链接(例如:<a href="jsp/login.jsp">)”,都是相对于base
的路径(即:http://localhost:8080/MyApp/),所以我们就可以进行的使用 相对链接,而不必担心,
转发操纵(forward)或 哀求地址不同不同所造成的页面无法找到的错误啦~(即: HTTP: 404)。。。
示例二、JSP页面提交表单给Servlet时
例如在web.xml中注册如下的servlet:
<servlet>
<servlet-name>addStudent</servlet-name>
<servlet-class>org.mytest.addStudent</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>addStudent</servlet-name>
<url-pattern>/servlet/addStudent</url-pattern>
</servlet-mapping>
如果说,你工程名字为HibernateApp3,JSP页面提交表单给servlet时有两种写法,表单在web项目下:
1. <form action=servlet/addStudent method=post>...</form>
2. <form action=/HibernateApp3/servlet/addStudentmethod=post>...</form>
留意:由于不是在服务器,所以/代表根目录(即使Tomacat根目录),假如路径是使用/开头,Tomcat就是webApp那个目录,
假如你不是/开头代表你从当前工程的目录开始,就是相对路径,例如:webApp/HibernateApp3/
这一点非常重要,良多提交表单时发生的错误都是由于提交路径犯错造成的。 |
|