JavaWEB路径总结

 

  这篇文章是小编一直想写的一篇,主要是对web阶段中各个路径进行的一些总结,希望读者看过之后对于路径方面有一个清晰的认识。首先声明一点:世界上一切东西都是相对的,对于这点而言,相信大家并不陌生,从初中开始我们就接触了物体的相对运动,由于这篇文章是针对于WEB阶段来讲的,所以以下绝对路径和相对路径都是针对于整个互联网而言的。

客户端路径

       是指运行在浏览器上的路径。比如:表单、超链接、js(location.href)、Ajax(url)、CSS和JS的引入以及重定向等。路径分为绝对路径和相对路径,相对路径又分为相对主机的路径和相对于当前请求的路径。干说没用,举例如下:

  例1:假设a.html的路径为: http://localhost:8080/hello/pages/a.html 。a.html的内容如下:

<a href="http://localhost:8080/hello/pages/b.html">绝对路径</a>

<a href="/hello/pages/b.html">相对于主机的路径</a>

<a href="b.html">相对于请求的路径</a>

  绝对路径(以协议开头的路径):最终请求路径为:http://localhost:8080/hello/pages/b.html

  相对路径:

  相对于主机的路径(以“/”开头):相对于当前主机(可以简单理解为ip地址,如果想深入了解请研究tomcat的Server.xml文件。这里是localhost)的路径,请求的最终路径为:http://localhost:8080/hello/pages/b.html

        相对于请求的路径(不以“/”开头):相对于当前请求(浏览器的请求)的路径,如果是直接访问的a.html页面,则当前请求为:http://localhost:8080/hello/pages/a.html .最终请求路径为:http://localhost:8080/hello/pages/b.html ;如果是请求的http://localhost:8080/hello/aServlet   然后在服务器转发到了a.html页面。则当前请求为:http://localhost:8080/hello/aServlet,请求的最终路径为:http://localhost:8080/hello/b.html

 

        例2:假设当前请求为http://localhost:8080/hello/aServlet。当前请求的Servlet(或Controller)中有如下重定向的代码:

//绝对路径
response.sendRedirect("http:localhost:8080/hello/page/b.html");

//以“/”开头的相对路径
response.sendRedirect("/hello/page/b.html");

//不以“/”开头的相对路径
response.sendRedirect("b.html");

  绝对路径是以协议开头的,没什么可说的。

  以“/”开头的相对路径(相对于当前主机的路径):请求的最终路径为:http://localhost:8080/hello/pages/b.html

  不以“/”开头的路径(相对于当前请求的路径):请求的最终路径为:http://localhost:8080/hello/b.html

  注:

    1.  推荐使用相对于主机的路径。这样跟当前请求无关。

  2. 重定向是在服务器上设置了302的状态码和location的响应头。浏览器接收之后按照location的响应头重新发送请求,由于重定向的地址是在客户端重新请求的地址,所以重定向属于客户端路径。

服务端路径

  服务端路径是指在服务器上面运行的请求,比如请求转发(常用)、请求包含等。服务端的路径有两种:相对于当前应用的路径和相对于当前请求的路径。

  例:假设当前请求的路径为:http://localhost:8080/hello/servlet/aServlet ,在后台有如下代码:

//以“/”开头,相对于当前应用的路径
request.getRequestDispatcher("/servlet/b.html").forward(request, response);

//不以“/”开头,相对于当前请求的路径
request.getRequestDispatcher("b.html").forward(request, response);

  相对于当前应用的路径(以“/”开头):最终转发的地址为:http://localhost:8080/hello/servlet/b.html 。

  相对于当前请求的路径(不以“/”开头):最终转发的地址为:http://localhost:8080/hello/servlet/b.html 。

  另:所有Web层框架的底层使用的都是Web的基础Filter(Struts)或Servlet(SpringMVC),请求都是request,响应都是response,所以各个Web层框架的转发或重定向底层都是利用request和response进行的。  

<url-pattern>路径

  这里的路径仅仅是指web.xml文件中的标签内容。这个路径是虚拟的路径,只有相对路径(相对于当前应用的路径),但相对路径的写法可以是精确查询和模糊查询。

       例:

<!—精确查询,是相对于当前应用-->
<url-pattern>/servlet/testPath</url-pattern>

<!—模糊查询,表示匹配servlet目录下的所有文件或请求。/*表示匹配所有-->
<url-pattern>/servlet/*</url-pattern>

<!—模糊查询,表示匹配所有后缀名为do的文件或请求-->
<url-pattern>*.do</url-pattern>

  在web.xml文件中通配符(*)只能放在路径开头或结尾,不能放在中间。如“/*.do”是非法的。

  注:tomcat查找路径的顺序:优先具体的路径,其次是匹配目录,然后是后缀名。最后是缺省路径。

获取资源的路径

   获取资源的路径主要有3种,分别是ServletContext、Class和ClassLoader。其中ServletContext是WEB阶段的,Tomcat提供的一种获取资源的方式;Class和ClassLoader获取资源主要是JavaAPI提供的一种获取流的方式,由于这是JDK提供的,所以不仅局限于Web,在普通Java类中也可以使用,主要用于获取src目录及其子目录下的文件流。

ServletContext获取资源的路径。

  这里ServletContext获取资源的路径是相对系统的绝对路径(在Windows中是带盘符的,可以用来获取上传或下载文件的具体路径)。

  基本语法:servletContext.getRealPath("路径");参数中的路径必须是相对路径,可以“/”开头,也可以不使用“/”开头,但无论是否使用“/”开头都是相对当前应用路径,建议以"/"开头(这样可以尽量统一)。

  另外获取ServletContext的方法如下:

  1. 使用request获取: request.getSession().getServletContext();
  2. 在Servlet中获取:this.getServletContext();
  3. 使用FilterConfig对象获取(在Filter中使用):config.getServletContext();

  示例如下:假如当前tomcat的路径为:D:\SoftWare\development\tomcat7.0.52,项目名为hello。有如下代码:

ServletContext servletContext = this.getServletContext();

String path1 = servletContext.getRealPath("a.txt");

String path2 = servletContext.getRealPath("/a.txt");

  则path1和path2一样:D:\SoftWare\development\tomcat7.0.52\webapps\hello\a.txt

  所代表的网络路径为:http://localhost:8080/hello/a.txt

Class获取资源

  Class获取资源主要是用作自己写的配置文件,用来读取内容。

  用法:clazz.getResourceAsStream("路径")。参数中的路径可以以“/”开头,也可以不以“/”开头。其中带“/”的表示相对于当前类的路径,不以“/”开头表示相对于当前class所在目录的路径。

  示例:假设当前类的包结构为:com.wind-snow.test.path

//获取Class对象
Class clazz = this.getClass();
//带“/”的路径(相对于当前的类路径)
InputStream stream1 = clazz.getResourceAsStream("/a.txt");
//不带“/”的路径(相对于当前class所在目录的路径)
InputStream stream2 = clazz.getResourceAsStream("a.txt");

  stream1代表src目录下的a.txt文件,发布之后代表“/项目名/WEB-INF/classes/a.txt”文件。

  stream2代表src目录下的com/wind-snow/test/path/a.txt文件,发布之后代表“/项目名/WEB-INF/classes/com/wind-snow/test/path/a.txt”文件。

   注:

    1. 若找不到。此方法不会报异常。只是会返回一个null;。

    2. 这个方法的底层使用的是ClassLoader的getResourceAsStream方法。

ClassLoader获取资源

  和Class获取资源类似。只是不同的写法而已,也是用于获取文件流的。

  用法:classLoader.getResourceAsStream("路径")。参数中的路径可以以“/”开头,也可以不以“/”开头(建议)。但带不带“/”的都表示相对于当前类的路径。

  示例:假设当前类的路径为:com.wind-snow.test.path

ClassLoader classLoader= this.getClass().getClassLoader();

InputStream stream1 = classLoader.getResourceAsStream("/a.txt");

InputStream stream2 = classLoader.getResourceAsStream("a.txt");

 

  stream1和stream2都代表src目录下的a.txt文件,发布之后代表“/项目名/WEB-INF/classes/a.txt”文件。

   注:若找不到。此方法不会报异常。只是会返回一个null。

总结:

        1. 客户端是带“/”相对当前主机。

        2. 服务端(包括上述的服务端、url-pattern路径和ServletContext路径)带不带“/”都是相对当前应用(建议带上)。

        3. Class带“/”是相对当前类路径。不带“/”是相对于当前位置。

        4. ClassLoader无论带不带“/”都是当前类路径(建议带上)。

 

  另附:总结的一张图片如下,此图是使用XMind画的,原件下载地址:https://files.cnblogs.com/files/wind-snow/Web路径总结.zip

posted @ 2017-09-26 12:59  风中抚雪  阅读(765)  评论(0编辑  收藏  举报