java 包管理

JVM 的工作流程

  1. 执行一个类的字节码

  2. 执行过程中碰到新的类,搜索新的类,并加载它

  3. 重复 1 和 2

问:那 JVM 从哪里寻找并加载类呢? 答:在 Classpath 中寻找

举个例子,如下我只是构建了一个非常简单的打印 hello 的 Java 程序:

package com.github.hcsp.objectbasic;

public class Print {
    public static void main(String[] args) {
        String s = "Hello";
        System.out.println(s);
        s = s.toUpperCase();
        System.out.println(s);
    }
}

当我们在 idea 中按下 Run 时,idea 终端会显示如下信息:

"C:\Program Files\Java\jdk1.8.0_311\bin\java.exe" . . .
Hello
HELLO

Process finished with exit code 0

我们展开"C:\Program Files\Java\jdk1.8.0_311\bin\java.exe" . . .

问题来了,我们只是运行了一个类,classpath 就已经是一长串引用了,当项目复杂的时候,所引用的类库里还会引用别的类库,层层引 用就会陷入类似回调地狱那样的状态,而且还有一个问题就是同样包的不同版本会被不同的库引用,这时我们应该引用哪个版本呢 ?maven 的出现让这些问题得到极大的改善。

maven

  1. 约定大于配置,例如默认的 src/main src/test 目录

  2. maven 中央仓库,按照 groupId/artifactId/version 的约定为所有包编号

  3. maven 本地仓库放在~/.m2 文件夹中

  4. 解决传递性依赖问题,不允许同名不同版本的包同时出现,依赖冲突就近选择

在命令行中使用mvn dependency:tree查看解决包冲突后的依赖树,maven 依赖冲突时使用就近选择方式 例如如下 ABCD 四个包的依赖关系: A0.1>B0.2>C0.2 D0.2>C0.1 maven 最终会选择加载 C0.1 这个包,这就会出现新的问题,比如 C0.1 版本里缺失 C0.2 中新增的方法,运行时就会报错。

  1. 解决方法 1:直接在 pom.xml 中引入你想要的版本 直接引入我想要的 C0.2 版本

  1. 解决方法 2:排除某个包的某个依赖(idea 中可以使用 maven helper 插件)

maven 的 pom.xml 的 dependency 中的 scope 属性的作用

scope 的取值:

  1. test,仅在测试环境加载当前包

  2. compile,编译和运行时都加载当前包

  3. provided,仅在编译的时候使用当前包

Last updated

Was this helpful?