http://swiftlet.net/archives/713
Java中不存在标准的相对路径,各种相对路径取资源的方式都是基于某种规则转化为绝对路径。所以在Java中文件路径问题无非归结为一点:找基点,也就是在某种环境下(web、j2ee或jar包等)通过合适的方式找到一个稳定的基点,然后通过这个基点找到你要的文件或资源。Java中的基点有哪些呢?大致总结一下有以下几种:
(1)classpath
如果你要找的资源在classpath下,那么通过classpath这个基点是比较合适的,而取得这个基点方式主要是通过ClassLoader来,具体方法就是ClassLoader.getResource(String name),而取得ClassLoader的方式很多,比如:
Thread.currentThread().getContextClassLoader()
clazz.getClassLoader()
ClassLoader. getSystemClassLoader()
ClassLoader找resource的实现原理就是先递归在parent classLoader中,从其所在classpath里加载resource,如果所有层级的classLoader都未找到,则调用此classLoader的findResource方法来找,而这个方法是暴露给自制classLoader来现实的,因此给了在classpath之外加载resource的机会。
(2) 当前用户目录
就是相对于System.getProperty("user.dir")返回的路径, 对于一般项目,这是项目的根路径,对于JavaEE服务器,这可能是服务器的某个路径。这个并没有统一的规范!然而,默认情况下,java.io 包中的类总是根据当前用户目录来分析相对路径名,如new File("xxx"),就是在 System.getProperty("user.dir")路径下找xxx文件。因此,通过这种方式来定位文件可能会出现移植问题。
(3) Web应用程序的根目录
在Web应用程序中,我们一般通过ServletContext.getRealPath("/")方法得到Web应用程序的根目录的绝对路径。掌握了上面几个基点,就能很轻松得定位你要找的resource。
Class.getResource()与Class.getResourceAsStream()方法,但很多人还是不太懂它的用法,下面给还不是很熟的人解释一下。 比如我们有以下目录
|--project
|--src
|--package
|--Test.java
|--file1.txt
|--file2.txt
|--bin
|--package
|--Test.class
|--file3.txt
|--file4.txt
在上面的目录中,有一个src目录,这是JAVA源文件的目录,有一个bin目录,这是JAVA编译后文件(.class文件等)的存放目录。那么,我们在Test类中应该如何分别获得file1.txt file2.txt file3.txt file4.txt这四个文件呢?
首先讲file3.txt与file4.txt
file3.txt:
方法一:File file3 = new File(Test.class.getResource("file3.txt").getFile());
方法二:File file3 = new File(Test.class.getResource("/package/file3.txt").getFile());
方法三:File file3 = new File(Test.class.getClassLoader().getResource("package/file3.txt").getFile());
备注:方法三与方法二之间的区别是没有前面的"/","/"这个代表“class文件的根目录”。
file4.txt:
方法一:File file4 = new File(Test.class.getResource("/file4.txt").getFile());
方法二:File file4 = new File(Test.class.getClassLoader().getResource("file4.txt").getFile());
对于file3.txt与file4.txt,我们可以有多种方法选择,但是file1与file2文件呢?如何获得?
答案是,你只能写上它们的绝对路径,不能像file3与file4一样用class.getResource()这种方法获得,它们的获取方法如下。
假如整个project目录放在c:/下,那么file1与file2的获取方法分别为
file1.txt
方法一:File file1 = new File("c:/project/src/package/file1.txt");
file2.txt
方法一:File file2 = new File("c:/project/src/file2.txt");
总结一下,就是你想获得文件,你得从最终生成的.class文件为着手点,不要以.java文件的路径为出发点,因为真正使用的就是.class,不会拿个.java文件就使用,因为java是编译型语言。
至于getResouce()方法的参数,你以class为出发点,再结合相对路径的概念,就可以准确地定位资源文件了,至于它的根目录,你用不同的 IDE build出来是不同的位置下的,不过都是以顶层package作为根目录,比如在Web应用中,有一个WEB-INF的目录,WEB-INF目录里面除了web.xml文件外,还有一个classes目录,它就是你这个WEB应用的package的顶层目录,也是所有.class的根目录 "/",假如clasaes目录下面有一个file.txt文件,它的相对路径就是"/file.txt",如果相对路径不是以"/"开头,那么它就是相对于.class的路径。
具体的实现过程见下面的代码分析:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
|
private String resolveName(String name) {
if (name == null) {
return name;
}
if (!name.startsWith("/"))
{
Class c = this;
while (c.isArray())
{
c = c.getComponentType();
}
String baseName = c.getName();
int index = baseName.lastIndexOf('.');
if (index != -1)
{
name = baseName.substring(0, index).replace('.', '/')
+"/"+name;
}
} else
{
name = name.substring(1);
}
return name;
}
|
还有一个getResourceAsStream()方法,参数是与getResouce()方法是一样的,它相当于你用getResource()取得File文件后,再new InputStream(file)一样的结果。