首先来回顾一下java文件的执行:
假设有这样一个文件结构
在 root路径下,有com/a/A.class
现在在root路径的命令行下 执行
java A;
这样肯定会报错:找不到或无法加载主类,因为当前路径下没有A这个类啊!!
正确的执行方法:
java com.a.A
如果我要在非root路径下运行A类,怎么搞:
java -cp root com.a.A
也就是说用cp命令,将root路径引入到classpath中,这样,加载器加载com.a.A的时候,就会去我们传入的classpath中去寻找了。
这是classpath的简单用法。
当我们不手动指定classpath的时候,classpath就是当前路径,也就是执行java命令的地方。
搞清楚这些,再来说明资源路径引用的问题。
new File("a.txt");
new FileInputStream("a.txt");
如以上两行代码,用的都是相对路径,那么程序在运行的时候就会在当前程序运行的路径下,而不是在classpath中寻找文件(这一点很重要)。
再来看通过类和类加载器获取资源的方式:
1. 通过类获取资源
A.class.getResource("b.txt")
1) 如果是相对路径,会在当前类所在的路径下找,即 com.a下面
2) 如果以/开始,则从根路径去找
2. 通过类加载器获取资源
A.class.getClassLoader().getResource("a.txt")
会在classpath找,而classpath是可以在运行时传入的。例如
java -cp a/b/c A 那么类加载器也会在a/b/c路径下去找
注意,根据这个方法的API文档说明,其路径分隔符必须是/。
总结:
1. io流包括new File() 引用都是 从 程序运行的路径下找。
2. 通过类获取资源:会在类的包路径找
3. 通过类加载器获取资源:会在classpath中找