JavaWeb中读取文件资源的路径问题
javaweb中读取文件资源路径的问题
在做开发的时候,我们可能会需要从本地硬盘上读取某一个文件资源,或者修改某一个文件,这个时候就需要先找到这个文件,然后用FileInputStrem等文件字节。字符流来将这个文件读取到内存中,再对其进行修改等的操作。那么在找这个文件的过程中就涉及到一个路径的问题--->怎么正确的找到这个文件呢?
有两种可行的方法:一、绝对硬盘路径;二、相对路径。。来来来,我们掰扯一下这两种方法的不同-------
一、硬盘绝对路径
如图,我的E盘下面有个path,里面有一个1.txt的文件,,现在在Servlet中用绝对路径读取。
public void doPost(HttpServletRequest request, HttpServletResponse response)throws ServletException, IOException { String path = "E:\\path\\1.txt"; File file = new File(path); System.out.println(file.getAbsolutePath());//输出读取到的文件路径 }
访问这个servlet,结果读取正常
但是呢,这样写真的好吗???我们知道,一个web应用可能会存在一个更换服务器的情况,,,如果是一个上线的项目,一旦服务器换了,那么新换的服务器会存在E:\path\1.txt这个文件路径吗???答案:NO,所以---绝对路径虽然可以解决问题,但是很不利于维护及项目搬迁。
那么,,相对路径呢??
二、相对路径
所谓行对路径,,指的是相对于整个web应用在硬盘上的路径来说的,,来来来,我们一下。。。
首先,在我的web应用下的WebRoot下面有一个1.txt文件,,怎么读它呢?(注意啊,web应用都是发布到tomcat的webapps下面执行的)
可能我们会在servlet中这样写路径来读取:
哎哎哎,,等一下,,我要读的是web应用下得文件啊,,你丫怎么跑到tomcat/bin下面找了!!!!(我的tomcat/bin下面可没这个文件,,所以肯定读取不到)
那么,为什么会去tomcat/bin下面读呢?这就得说说Java文件读取的机制了:如果你直接这样写路径new File("a.***");那么tomcat就会从程序启动的地方去找这个文件(Java项目中也是如此),那么web应用是从有tomcat来执行的,,tomcat这个程序是从哪里启动的呢??答对了,就是tomcat/bin/startup.bat启动的。
so。。这样写肯定找不到的,,,那么怎么找??
方法1、servlet中用ServlertContext域的getRealPath()这个方法找
简单代码如下:
哎,这才对嘛,,我的项目是发布到tomcat的webapps文件夹下的,,这样才对嘛。。。所以是怎么一个原理呢?
对于ServletContext().getRealPath("路径名A");这个方法,无论你的路径名A是什么,ServletContext().getRealPath()方法都会在路径名A前拼上当前web应用的硬盘路径,,这样加上你传进去的路径就可以找成功找到了
小结一下:其实通过ServletContext().getRealPath来拼接路径只是绝对硬盘路径的升级版,,,但是它好久好在计时你的web应用换了服务器环境,只要你的文件在web应用的中的相对路径不变,那么不论你的web应用如何更换服务器环境,都能动态的获取当前服务器环境的绝对文件路径。
当然,上面的getRealPath()方法只能在中使用(因为只有servlet才有ServletContext域对象),在MVC模式中,很多时候servlet并不是来做处理请求的,跟多的是进行一个转发操作,那么,如果是在servlet中调用普通的Java类来读取文件呢???怎么读文件?
如果只是普通的类,那么你也可以直接使用绝对硬盘路径,,但同样,环境换了之后就可能会挂掉。可以通过类加载器ClassLoader类的getResource()方法来加载文件。。原理是:
类加载器是从你的web应用的WEB-INF\class下找.class文件来加载的。所以呢,你要找的web应用中的文件只要相对于你的class文件夹找就可以了。
来,我们try一try。。。新建一个普通的Java类PathofJava.java,
在中调用这个类的method()方法,访问一下这个servlet---
所以正确的方法是用类加载器去找:
为什么要用../../1.txt呢?因为1.txt做i中发布到tomcat中web应用下去了,跟index.jsp平级,所以用../(表示上级目录)来找
小结一下:其实类加载器也是搞出来一个绝对硬盘路径来找文件的,只是这个绝对路径是随着web应用的路径变化而变化的,就不存在web应用换了环境找不到资源的问题了
附上整个test项目的结构
Java Web开发中路径问题小结
(1) Web开发中路径的几个基本概念
假设在浏览器中访问了如下的页面,如图1所示:
图1
Eclipse中目录结构如图2所示:
图2
那么针对这个站点的几个基本概念表述如下:
1. web站点的根目录:http://localhost:8080/
2. web应用程序的的根目录:http://localhost:8080/test/
3.同级目录:http://localhost:8080/test/articles/article1.jsp和http://localhost:8080/test/articles/article2.jsp 就是处于同级目录。
二.几个路径相关函数的返回值:
在TestURL.jsp文件中测试了三个路径相关函数:
1. getRequestURI()
2. getContextPath()
3. getServletPath()
结果如图3所示:
图3
三.web.xml中的路径
Servlet 映射 <url-pattern>/xx</url-pattern> 中的 “/” 代表当前 web 应用的根路径。
这个只要思考一下就能得出结论:所有的web.xml文件都是描述某个web应用的部署相关信息,所以“/”只能代表当前 web 应用的根路径,而不是指向web站点的根目录
四.转发和重定向中的路径问题
1. 如果没有“/”,如request.getRequestDispatcher("hello.jsp ").forward(request, response) 和response.sendRedirect(" hello.jsp ") 就表示在同级目录中寻找login.jsp文件。
2. 如果路径中包含“/”(注意,这里的“/”是指路径中的第一个“/”)
a, 转发
如request.getRequestDispatcher("/hello.jsp").forward(request, response) :代表到http://localhost:8080/test/ 下目录寻找hello.jsp文件。
b,重定向
如response.sendRedirect("/hello.jsp"):代表到http://localhost:8080/ 目录下寻找hello.jsp文件
五.<form action=“/xxx”> 或 <a href=“/xxx”></a>中的路径
“/”代表的是 web 站点的根路径.。
因为超链接可以链接到任何需要的目标资源, 所以 / 代表的肯定不是当前 web 应用的根路径, 而是当前 web 站点的根路径。
六.Page标签中errorPage属性的路径问题
errorPage属性值如果以”/”开头表示,表示相对于当前WEB应用程序的根目录,否则,表示相对与当前页面。
Java Web中资源的访问路径
在web应用中,以“/”开头的是绝对路径,不以“/”开头的是相对路径。在服务器端,通常都使用绝对路径。例如web.xml、struts.xml、servlet等的访问路径都是以“/”开始。
服务器端的“/”等同于:http://localhost:8080/day01/。(day01为工程名)
在客户端,可使用绝对路径,也可使用相对路径。例如html、jsp、css、javascript中。
但是客户端的绝对路径与服务器端的绝对路径中的“/”代表的意思是不同的。
客户端的“/”等同于:http://localhost:8080/。(它是不带工程名的)
比如:这里有一个web工程:day01,在工程根目录下有两个页面:1.jsp和2.jsp。
1.jsp中有一个超链接:<a href=”http://m.cnblogs.com/161445/“/2.jsp”>点击跳转到2.jsp</a>,我们在地址栏输入:http://localhost:8080/day01/1.jsp即可
访问1.jsp,然后点击超链接后提示错误404,找不到页面。这是由于我们点击超链接后的访问地址是这样的:http://localhost:8080/2.jsp
所以提示找不到。超链接改成这样就可以了:<a href=”http://m.cnblogs.com/161445/“/day01/2.jsp”>点击跳转</a>
如果使用相对路径是怎样的呢?
比如:这里有一个web工程:day01,在工程根目录下有一个页面:1.jsp,同时有一个目录:jsp,jsp目录下有一个目录:menu,menu目录下有一个页面:2.jsp。我们在地址栏输入: http://localhost:8080/day01/jsp/menu/2.jsp 可访问到2.jsp。
在2.jsp中有一个超链接,点击可跳转到1.jsp。假如这个超链接是这样的:<a href=”http://m.cnblogs.com/161445/“1.jsp”>点击跳转到1.jsp</a>,点击后提示错误404,找不到页面。因为这里使用了相对路径(相对与当前页面),点击超链接后的访问路径是这样的:http://localhost:8080/day01/jsp/menu/1.jsp,所以提示找不到。改成这样就可以了:<a” target=”_blank” rel=”nofollow”>
可以看出,客户端使用相对路径非常麻烦,需要考虑当前页面的位置,所以在客户端我们也统一使用绝对路径。
但是客户端的绝对路径中都要带着工程名,是写死的,假如有一天工程名修改了,所有页面中的超链接都要改,工作量太大。
所以这里的工程名不能写死,要动态获取。改成这样就可以了:<a href=”http://m.cnblogs.com/161445/“<%25=request.getContextPath()” target=”_blank” rel=”nofollow”>
request.getContextPath()就是获取工程名:/day01