在Java Web中相对路径和绝对路径区别及应用方法

闲来无事,研究了一下bootstrap框架,并把这个框架融入到我开发的一个项目中,在应用过程中发现了一些问题,经过潜心研究这个问题终于解决了,下面我就把整个过程分享给大家。

一、开发环境介绍

开发语言   JDK1.8
IDE ECLIPSE platform 4.6.0
WEB容器 

tomcat9

后端框架 Spring+Spring MVC+Spring Data
前端框架 bootstrap3.3.7

二、问题回顾

1、项目说明

 

2、static.html文件(改成jsp文件也可以)

 注意:红色框内容<link rel="stylesheet" href="css/bootstrap.min.css"/>,采用了相对路径

运行效果:

现象一:

扫描二维码关注公众号,回复: 2022175 查看本文章

 现象二:

CSS样式表文件没有起作用。查看源代码,点击CSS连接,如下图:

三、解决方案

1、针对“现象一”的解决方案

在/OutSourcingManage/WebContent/WEB-INF/dispatcherServlet-servlet.xml文件中加入如下语句:

<mvc:default-servlet-handler/>

2、针对“现象二”的解决方案

方法一、在/OutSourcingManage/WebContent/WEB-INF/dispatcherServlet-servlet.xml文件中加入如下语句:

<mvc:resources mapping="/css/**" location="/bootstrap/css/"/>

方法二、修改static.html,将“<link rel="stylesheet" href="css/bootstrap.min.css"/>”修改为:

<link rel="stylesheet" href="/OutSourcingManage/bootstrap/css/bootstrap.min.css"/>

方法三、将static.html修改static.jsp,并增加如下语句:

<%@ page language="java" contentType="text/html; charset=UTF-8"

pageEncoding="UTF-8"%>
<%
String path = request.getContextPath();
String basePath = request.getScheme()+"://"+request.getServerName()+":"+request.getServerPort()+path+"/";
%>

<head>

<base href="<%=basePath%>">

....

四、原理剖析

1、目标效果

2、内部原理剖析

(1)静态资源的处理

 

Spring MVC的拦截器DispatcherServlet默认是拦截所有请求,包括*.js、*.css、*.html、*.jpg、*.jsp等静态资源和动态资源,但是静态资源没有对应的handler,所以会出现上述警告。为了解决这个问题,需要为*.js、*.css、*.html、*.jpg等静态资源配置tomcat默认的拦截器。而<mvc:default-servlet-handler/>就是为此目的而生。

(2)绝对路径、相对路径

关于绝对路径、相对路径对于很多初学者感到迷惑,即使写出了可以运行的程序,也没有真正搞清楚原理,甚至对某些老手,也偶尔会犯迷糊。我总结了一些认识规则,与各位分享。

  • URL的一般语法格式为:
URL由三部分组成:资源类型、存放资源的主机域名、资源文件名。
(带方括号[]的为可选项):protocol:// hostname[:port] / path / [;parameters][?query]#fragment
protocol(协议):一般是ftp、file、http(https)、mailto、thunder等协议。
hostname(主机名):是指存放资源的服务器的DNS(域名服务器)主机名或IP地址。有时,在主机名前也可以包含连接到服务器所需的用户名和密码(格式:username:password@hostname)。
port(端口号):各种传输协议都有默认的端口号,如http的默认端口为80,我们在开发过程中一般采用8080。
path(路径):由零或多个“/”符号隔开的字符串,一般用来表示主机上的一个目录或文件地址。
parameters(参数):这是用于指定特殊参数的可选项。
query(查询):用于传递参数,可有多个参数,用“&”符号隔开,每个参数的名和值用“=”符号隔开。
fragment(信息片段):用于指定网络资源中的片段。
以http请求为例,URL地址是http://192.168.0.1:8080/OutSourcingManage/login?username=gaodianhua&password=123456
  • URL格式细分含义

http:/主机名:[端口号]/web应用根目录/资源名称[?query]。

下面以http://192.168.0.1:8080/OutSourcingManage/login?username=gaodianhua&password=123456为例

①服务器根地址(或称为tomcat服务器地址)规范为:

http:/主机名:[端口号]/,例如:http://192.168.0.1:8080/

②web应用根地址规范为:

http:/主机名:[端口号]/web应用根目录/,例如:http://192.168.0.1:8080/OutSourcingManage/

③资源根地址规范为:

http:/主机名:[端口号]/web应用根目录/资源名称,例如:http://192.168.0.1:8080/OutSourcingManage/login

请记住:上面三个概念,即:tomcat服务器根地址、web应用根地址、资源根地址 

  • 服务器端(tomcat)和客户端(浏览器)对URL地址的解析规则

对于请求的任何资源,均需要转化为绝对URL地址。

服务器端对URL地址的解析一般使用web应用根地址作为相对路径。例如页面跳转和重定向时使用的地址:

request.getRequestDispatcher("/WEB-INF/views/success.jsp").forward(request, res);//页面跳转,跳转成功

request.getRequestDispatcher("WEB-INF/views/success.jsp").forward(request, res);//页面跳转,跳转成功

response.sendRedirect("/hello.html");//重定向,重定向失败

response.sendRedirect("hello.html");//重定向,重定向成功

"/"在服务器端和客户端的处理规则是不同的,即:

  1. 在服务器端解析时,被认为是以web应用根地址为基准地址的相对路径。
  2. 在客户端解析时,被认为是tomcat服务器根地址。

相对路径即不带"/"时,在服务器端和客户端的处理规则则是相同的,即:

  1. 在服务器端解析时,被认为是web应用根的地址。
  2. 在客户端解析时,被认为是父页面请求时web应用根地址为基准地址的相对地址。

下面以hello.html测试可得:

<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<title>Insert title here</title>
</head>
<body>
<H1>Hello World</H1>
<a href="world.html">测试相对地址</a>
<a href="/world.html">测试绝对地址</a>
</body>
</html>

注:world.html和hello.html在相同目录下。

地址栏输入:http://localhost:8080/OutSourcingManage/hello.html,其中,"http://localhost:8080/OutSourcingManage/"为web应用根地址。

点击"测试相对地址"时,地址栏显示的是:http://localhost:8080/OutSourcingManage/world.html(请求成功)

点击"测试绝对地址"时,地址栏显示的是:http://localhost:8080/world.html(请求失败,此时要想成功需要手工补充web应用地址)

如果地址栏输入:http://localhost:8080/OutSourcingManage/springmvc/hello.html,其中,"http://localhost:8080/OutSourcingManage/"为web应用根地址。

点击"测试相对地址"时,地址栏显示的是:http://localhost:8080/OutSourcingManage/springmvc/world.html(请求成功)

点击"测试绝对地址"时,地址栏显示的是:http://localhost:8080/world.html(请求失败,此时要想成功需要手工补充web应用地址)

注意:①页面跳转(forward)是以web应用根地址为基准的,可以跳转到同应用下任意有效资源页面。

         ②WEB-INF只能被服务器端访问,不能被客户端访问。

         ③重定向(sendRedirect)是服务器向客户端响应的资源名称,由客户端重新向服务器端请求。

         ④重定向可以跨web应用。

         ⑤基准路径或者叫相对路径的基准:生手开车时,经常出现会问一个问题:"我的方向盘向左方(或向右方)打了半圈,还是一圈,还是打死了?",教练常常会告诉我,如果记不清了就把方向盘打死,然后回一圈半,方向盘就正了。同样,在操作系统中,可以使用命令"pwd"查看当前所处的目录。这些告诉我们基准路径很重要。

(3)一点建议

由于在日常开发过程中,相对地址维护起来非常复杂,容易混乱,建议统一采用绝对地址。在html/jsp中有<base>标签。利用该标签可以实现绝对地址访问。以JSP文件为例:(注意红色字体部分)

<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%>
<%
String path = request.getContextPath();
String basePath = request.getScheme()+"://"+request.getServerName()+":"+request.getServerPort()+path+"/";
%>
<!DOCTYPE html>
<html>
<head>

      <base href="<%=basePath%>">

  <meta http-equiv="Content-Type" content="text/html; charset=utf-8">
  <title>Insert title here</title>
  <link rel="stylesheet" href="css/bootstrap.min.css"/>
</head>

<body>

 ....

五:总结

     终于写完了,整整一天时间(国庆节第二天),虽有感觉有点累,但一想到知识传播会帮助很多初学者,我心里又是非常的高兴。自己鼓励一下自己吧。努力,加油。

分享是一种快乐,更是一种精神,我们生活在一个知识爆炸的社会,知识每天日新月异,我崇尚“工匠精神”,更喜欢“工程师文化”,把分享当作对自己的一种动力和督促,让自己不放松、不舒适,每天都生活积极向上的进步中和对工作、生活的思考中,永不懈怠。一句话,我们永远在路上,在筑梦的路上。 我非常注重学习知识的基础原理,探究系统背后的运作原理,运用知识体系化的学习方法,解决看似复杂的问题。有些问题就是那么简单,但其背后的原理却非常复杂,这就是为什么要学习的原因。学习不是为了“我会了,我懂了”,而是探究事物的本质和事物之间的联系,提升自己的学习能力。我把一个人的知识体系看作一个“黑洞”,只有这个“黑洞”具有足够大的质量和密度,才能源源不断的吸收知识,学习速度和能力才能不断提升。中国有句俗语“厚积薄发”,我想道理是相通的。 与各位有着同样理念的小伙伴们共勉。

猜你喜欢

转载自www.linuxidc.com/Linux/2016-10/135693.htm