java在操作文件的时候必然需要知道这个文件在哪里。每种文件就是一种资源,java获取资源的方式有两种:
1.通过java.io包中的类,例如File,其他类和File类似
2.通过Class或者ClassLoader
先看看File类:
An abstract representation of file and directory pathnames.
User interfaces and operating systems use system-dependent pathname strings to name files and directories. This class presents an abstract, system-independent view of hierarchical pathnames.
An abstract pathname has two components:
An optional system-dependent prefix string, such as a disk-drive specifier, "/" for the UNIX root directory, or "\\\\" for a Microsoft Windows UNC pathname, and
A sequence of zero or more string names.
官方文档说File类是文件和目录名的一种抽象代表,不依赖于系统,注意理解这句话,在下面的测试代码中会有所体现
import java.io.File;
import java.io.IOException;
public class TestResourcePath {
public static void main(String[] args) throws IOException {
TestResourcePath t = new TestResourcePath();
t.testFile();
t.testClass();
t.testClassLoader();
}
public void testFile(){
File file=new File("dir/1.txt");//注意执行该句,即使这个文件不存在,这个文件也不会被创建,只是File类的对象被构造了,这个文件并不会产生
//file.createNewFile();//File无法递归的创建文件及文件夹
System.out.println(file.getAbsolutePath());
File file2=new File("/dir/1.txt");//绝对路径
System.out.println(file2.getAbsolutePath());
File file3=new File("1.txt");
try {
file3.createNewFile();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}//可行
System.out.println(file3.getAbsolutePath());
File file4=new File("2.txt");//相对路劲
System.out.println(file4.getAbsolutePath());
File file5=new File("");
System.out.println(file5.getAbsolutePath());
File file6=new File("RC/0.gif");
System.out.println(file6.getAbsolutePath());
File file7=new File("/RC/0.gif");//绝对路劲
System.out.println(file7.getAbsolutePath());
System.out.println(System.getProperty("user.dir"));
System.out.println();
}
public void testClass(){
System.out.println(this.getClass().getResource("RC/0.gif"));
System.out.println(this.getClass().getResource("/RC/0.gif"));
System.out.println(this.getClass().getResource("1.txt"));
System.out.println(this.getClass().getResource("/1.txt"));
System.out.println(this.getClass().getResource(""));
System.out.println(System.getProperty("java.class.path"));
System.out.println();
}
public void testClassLoader(){
System.out.println(this.getClass().getClassLoader().getResource("RC/0.gif"));
System.out.println(this.getClass().getClassLoader().getResource("/RC/0.gif"));
System.out.println(this.getClass().getClassLoader().getResource("1.txt"));
System.out.println(this.getClass().getClassLoader().getResource("/1.txt"));
System.out.println(this.getClass().getResource(""));
System.out.println(System.getProperty("java.class.path"));
System.out.println();
}
}
下面是打印结果:
E:\Eclipse\javaStudyProjects\Reflect\dir\1.txt
E:\dir\1.txt
E:\Eclipse\javaStudyProjects\Reflect\1.txt
E:\Eclipse\javaStudyProjects\Reflect\2.txt
E:\Eclipse\javaStudyProjects\Reflect
E:\Eclipse\javaStudyProjects\Reflect\RC\0.gif
E:\RC\0.gif
E:\Eclipse\javaStudyProjects\Reflect
这几个文件程序执行之前只有0.gif是存在的,执行之后只多了一个E:\Eclipse\javaStudyProjects\Reflect\1.txt;也即是说new File 时,即使这个文件不存在,这个文件也不会被创建,只是File类的对象被构造了,这个文件并不会产生,印证了An
abstract representation of file and directory pathnames.File类只是一个抽象代表,并不是正真的文件,要想这个文件创建,需要调用createNewFile()方法。如果文件已存在,createNewFile()方法返回假,不会再次创建。
关于pathname的写法:
加“/”前缀的表示的是绝对路径,看file2和file7
不加前缀的是相对路径,并且这个相对是相对于E:\Eclipse\javaStudyProjects\Reflect\这个路径,即user.dir
再来看看Class
打印结果:
file:/E:/Eclipse/javaStudyProjects/Reflect/bin/RC/0.gif
file:/E:/Eclipse/javaStudyProjects/Reflect/bin/RC/0.gif
null
null
file:/E:/Eclipse/javaStudyProjects/Reflect/bin/
E:\Eclipse\javaStudyProjects\Reflect\bin
可以看出这个里面的前缀“/”不在起作用,因为/被过滤掉了,最后的结果都相当于相对路径,并且这个相对路径是E:/Eclipse/javaStudyProjects/Reflect/bin/,即java.class.path;也即是当前运行的类所在包的顶层包所在路径;看看Class.getResource()源码
public java.net.URL getResource(String name) {
name = resolveName(name);
ClassLoader cl = getClassLoader();
if (cl==null) {
return ClassLoader.getSystemResource(name); // A system class.
}
return cl.getResource(name);
}
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;
}
在看ClassLoader:
打印结果:
file:/E:/Eclipse/javaStudyProjects/Reflect/bin/RC/0.gif
null
null
null
file:/E:/Eclipse/javaStudyProjects/Reflect/bin/
E:\Eclipse\javaStudyProjects\Reflect\bin
可以看出ClassLoader不处理前缀“/”,所以加/是错误的,返回null,其他都和Class一样,因为Class还是调用了ClassLoader的getResource() 方法