Java classpath 以及资源文件读取

默认的classpath

ORACLE documentation

The default class path is the current directory. Setting the CLASSPATH variable or using the -classpath command-line option overrides that default, so if you want to include the current directory in the search path, you must include "." in the new settings.

默认的classpath., 即当前的工作目录. 设置环境变量CLASSPATH或者使用java-classpath(或者-cp)参数会覆盖这一默认值. 如果希望包括.classpath中, 应该将其加入到.加入到新设置中. 如,

java -cp .:/another/path:another/path2 package.class.to.run

一般不要依赖于全局的CLASSPATH, 这样很容易导致修改环境变量或者将程序拷贝到其他电脑上后, 影响程序的运行.

Class.getResource()和ClassLoader.getResource()

使用

package path.classpath;
public class ClassPath {

    public static void main(String[] args) {
        System.out.println(System.getProperty("java.class.path"));
        System.out.println(ClassPath.class.getResource(""));
        System.out.println(ClassPath.class.getResource("/"));
        System.out.println(ClassPath.class.getClassLoader().getResource(""));
    }
}

运行:

java path.classpath.ClassPath

输出:

.
file:/home/smallfly/programming_projects/java/JavaLanguageFeatures/out/production/features/path/classpath/
file:/home/smallfly/programming_projects/java/JavaLanguageFeatures/out/production/features/
file:/home/smallfly/programming_projects/java/JavaLanguageFeatures/out/production/features/
null

可以看到Class.getResource的两种用法:

  1. 使用相对路径(不以/开头)
    那么根目录设定为这个类的class文件所在的目录下
  2. 使用绝对路径(以/开头)
    那么根目录设定为这个类的顶级package下. 比如说A.class的路径为App/classes/package1/package2/A.class, 那么跟目录就设定为App/classes/

ClassLoader.getResource用法:

  1. 相对路径
    跟目录设置在顶级package下
  2. 绝对路径
    不支持这种用法, 直接会返回null

原理

Class类的getResource方法:

    public java.net.URL getResource(String name) {
        name = resolveName(name);
        ClassLoader cl = getClassLoader0();
        if (cl==null) {
            // A system class.
            return ClassLoader.getSystemResource(name);
        }
        return cl.getResource(name);
    }

处理name参数的resolveName方法:

/**
* Add a package name prefix if the name is not absolute Remove leading "/"
* if name is absolute
*/
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;
    }

可以看到, 实际上Class类的getResource最终还是调用了ClassLoadergetResource方法.
resolveName方法中, 可以看到对name做了以下处理:

  • 文件名以/开头, 即绝对路径
    直接返回去掉开头的/字符的子字符串
  • 文件名是相对路径
    首先得到当前类的类名(全限定名称). 注意对数组类型的处理, 因为数组的类名在Java里面是比较奇怪的, 所以那个while循环是为了得到数组的元素类型.
    然后将该类的package中的.替换为目录分隔符/, 然后拼接传入的name.
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,958评论 19 139
  • 写java代码时常常需要加载一些外部的资源,通常我们会使用全路径名加载一份资源,比如:C:\Users\Yukai...
    德彪阅读 2,171评论 0 0
  • 无论是Servlet或者Spring、MyBatis,配置资源文件都是必不可少的一项工作,Java中主要提供了提供...
    FX_SKY阅读 2,648评论 0 2
  • ClassLoader翻译过来就是类加载器,普通的java开发者其实用到的不多,但对于某些框架开发者来说却非常常见...
    时待吾阅读 1,102评论 0 1
  • 属于今天的 被昨天占有 咆哮的我们 一直留在原地 没有走 嵇康酿的酒 流出一道沟 隔绝了 你我 浸染了 宇宙 气泡...
    H3190阅读 415评论 0 0