一文搞懂Java的ClassPath

一文搞懂Java的ClassPath,第1张

一文搞懂Java的ClassPath
  • 声明:本文是作者在独立思考和实验后发布,如有不正确的地方,请多指正。

  • 创建一个空的maven项目myBoot,编写一段测试代码

package com.lhb.rc;

public class DateTest {

    public static void main(String[] args) throws Exception{

        System.out.println("test");

    }

}

  • 运行结果

  • 查看idea执行的完整的java.exe
"C:Program FilesJavajdk1.8.0_221binjava.exe" "-javaagent:D:IEDINSjetbains2021jetbrains2021InsIntelliJ IDEA 2021.1.3libidea_rt.jar=57033:D:IEDINSjetbains2021jetbrains2021InsIntelliJ IDEA 2021.1.3bin" -Dfile.encoding=UTF-8 -classpath "C:Program FilesJavajdk1.8.0_221jrelibcharsets.jar;C:Program FilesJavajdk1.8.0_221jrelibdeploy.jar;C:Program FilesJavajdk1.8.0_221jrelibextaccess-bridge-64.jar;C:Program FilesJavajdk1.8.0_221jrelibextcldrdata.jar;C:Program FilesJavajdk1.8.0_221jrelibextdnsns.jar;C:Program FilesJavajdk1.8.0_221jrelibextjaccess.jar;C:Program FilesJavajdk1.8.0_221jrelibextjfxrt.jar;C:Program FilesJavajdk1.8.0_221jrelibextlocaledata.jar;C:Program FilesJavajdk1.8.0_221jrelibextnashorn.jar;C:Program FilesJavajdk1.8.0_221jrelibextsunec.jar;C:Program FilesJavajdk1.8.0_221jrelibextsunjce_provider.jar;C:Program FilesJavajdk1.8.0_221jrelibextsunmscapi.jar;C:Program FilesJavajdk1.8.0_221jrelibextsunpkcs11.jar;C:Program FilesJavajdk1.8.0_221jrelibextzipfs.jar;C:Program FilesJavajdk1.8.0_221jrelibjavaws.jar;C:Program FilesJavajdk1.8.0_221jrelibjce.jar;C:Program FilesJavajdk1.8.0_221jrelibjfr.jar;C:Program FilesJavajdk1.8.0_221jrelibjfxswt.jar;C:Program FilesJavajdk1.8.0_221jrelibjsse.jar;C:Program FilesJavajdk1.8.0_221jrelibmanagement-agent.jar;C:Program FilesJavajdk1.8.0_221jrelibplugin.jar;C:Program FilesJavajdk1.8.0_221jrelibresources.jar;C:Program FilesJavajdk1.8.0_221jrelibrt.jar;D:2022myMavenApptargetclasses" com.lhb.rc.DateTest

  • 除去jdk的jar包,将剩余的命令粘贴出来
"C:Program FilesJavajdk1.8.0_221binjava.exe"  -classpath "D:2022myMavenApptargetclasses" com.lhb.rc.DateTest

我们发现,执行这个命令也会正确输出结果。

  • 我们本项目的classes目录如下,该目录会作为一个classpath

这个命令带了一个-classpath参数,告诉JVM将-classpath参数后的路径作为类路径,而这个路径就是我们maven项目编译输出后的classes目录。而且,除了这个目录,我们上一条命令带的其他的jar包也算作了classpath,也就是说,classpath是有很多个。我们解压C:Program FilesJavajdk1.8.0_221jrelibrt.jar看一下


也就是说,rt.jar下的根目录,也会作为一个classpath。

验证一下多个classpath

  • 创建另外一个maven项目,写一个类

  • 然后将这个项目打包发布到本地仓库,并在myBoot项目中引入该依赖
   
        
            com.lhb.tc
            fastDate
            1.0
        
    
  • 然后我们尝试用类加载器去获取一下这个FastDate
package com.lhb.rc;

public class DateTest {

    public static void main(String[] args) throws Exception{

        System.out.println("test");
       System.out.println(DateTest.class.getClassLoader().loadClass("com.lhb.rc.utils.FastDate").newInstance());

    }

}


打印成功。

为什么当前项目也能使用类加载器加载其他包的类?是因为在执行java.exe 命令时也将fastdate这个jar包作为了classpath一部分。而Classloader默认就是从classpath下寻找类的。


测试加载classpath下的资源

  • 再myboot项目resource下新建一个user.properties
  • 尝试使用classloader加载该资源
package com.lhb.rc;

import com.lhb.rc.utils.FastDate;

import java.io.InputStream;
import java.util.Properties;

public class DateTest {

    public static void main(String[] args) throws Exception{

        System.out.println("test");
        System.out.println(DateTest.class.getClassLoader().loadClass("com.lhb.rc.utils.FastDate").newInstance());

        InputStream resourceAsStream = DateTest.class.getClassLoader().getResourceAsStream("user.properties");
        Properties properties = new Properties();
        properties.load(resourceAsStream);
        System.out.println(properties.getProperty("user"));
        System.out.println(properties.getProperty("age"));

    }

}

控制台输出也没问题。

  • 此时,我们在fastdate项目下新建一个userbk.properties文件,然后重新发布项目

  • 更新一下myboot项目依赖

  • 然后我们此时尝试读取classpath下userbk.properties,注意这个文件是在fastdate包下的哦。
package com.lhb.rc;

import com.lhb.rc.utils.FastDate;

import java.io.InputStream;
import java.util.Properties;

public class DateTest {

    public static void main(String[] args) throws Exception{

        System.out.println("test");
        System.out.println(DateTest.class.getClassLoader().loadClass("com.lhb.rc.utils.FastDate").newInstance());

        InputStream resourceAsStream = DateTest.class.getClassLoader().getResourceAsStream("userbk.properties");
        Properties properties = new Properties();
        properties.load(resourceAsStream);
        System.out.println(properties.getProperty("user"));
        System.out.println(properties.getProperty("age"));

    }

}

我们发现确实也能读取到其他classpath下的文件。

当多个classpath有相同的文件名会怎么样?

  • 将myboot项目的user.properties改名为userbk.properties

那么此时,多个classpath下相同的文件了,我门看看java读取的是哪一个。


我们发现,是读取的myboot项目下的userbk.properties

我们查看完成的java命令,发现是myboot项目的classpath在前,fastdate项目的classpath在后。

将这两个路径换一下再执行


我们发现,如果将fastdate包的classpath指定在前,那么该classpath下的userbk.properties。这个与先后加载的顺序有关。

欢迎分享,转载请注明来源:内存溢出

原文地址:https://54852.com/zaji/5596212.html

(0)
打赏 微信扫一扫微信扫一扫 支付宝扫一扫支付宝扫一扫
上一篇 2022-12-15
下一篇2022-12-15

发表评论

登录后才能评论

评论列表(0条)

    保存