一台机器下,多个Java版本的粗放与精细管理

2022-11-14 19:08:42 浏览数 (1)

前言

在软件开发过程中,经常会遇到“古老”的项目,这些项目的JDK还处于Java 6、Java 7甚至更早的版本。同时,在学习新的JDK特性时,往往又需要安装最新版本的JDK。鉴于这些情况,我们就需要在在本机环境中安装多个JDK,并且能够方便的切换。

本篇文章以Mac OS操作系统为例,演示一下如何安装多个JDK版本,并且进行切换。这里重点介绍两种方式,基于别名的形式和基于jenv软件的形式。

基于别名的JDK切换

这里以Mac操作系统下,Java 9为例进行演示。

JDK安装

下载Mac下Java 9的dmg安装包,按照步骤一路Next进行安装即可。

安装完成,在Mac的/Library/Java/JavaVirtualMachines目录下会出现两个(如果安装多个JDK则有多个)目录:

代码语言:javascript复制
jdk-9.0.4.jdk		jdk1.8.0_151.jdk

除了目前比较主流的Java 8之外,新的长期支持版本(LTS)有Java 11和Java 17,大家可根据需要进行按照。上面展示了有两个JDK版本的情况,其他版本对照即可。

环境变量配置

Mac下可通过bash_profile文件来对JDK的环境变量进行配置。执行以下命令打开配置文件:

代码语言:javascript复制
vim ~/.bash_profile

如果原本没有.bash_profile文件,在运行vim ~/.bash_profile命令时会创建该文件。

参考原有环境变量配置,新增JDK9的配置:

代码语言:javascript复制
# Java config
export JAVA_8_HOME="/Library/Java/JavaVirtualMachines/jdk1.8.0_151.jdk/Contents/Home"
export JAVA_9_HOME="/Library/Java/JavaVirtualMachines/jdk-9.0.4.jdk/Contents/Home"

# config alias
alias jdk8="export JAVA_HOME=$JAVA_8_HOME"
alias jdk9="export JAVA_HOME=$JAVA_9_HOME"

# config default jdk
export JAVA_HOME=$JAVA_8_HOME
export PATH="$JAVA_HOME:$PATH"

上面修改了两处:第一处定义了JAVA_8_HOME和JAVA_9_HOME的Home路径;第二处配置了jdk8和jdk9的操作的别名(alias)。

保存上述修改配置,并执行以下命令使bash_profile生效:

代码语言:javascript复制
source ~/.bash_profile

编译完成,相关的配置即完成,后续可通过命令来进行JDK的切换。

JDK环境切换

在上述配置中,默认使用的是Java 8,执行java -version命令可验证:

代码语言:javascript复制
192:JavaVirtualMachines zzs$ java -version
java version "1.8.0_151"
Java(TM) SE Runtime Environment (build 1.8.0_151-b12)
Java HotSpot(TM) 64-Bit Server VM (build 25.151-b12, mixed mode)

当需要切换到Java 9时,执行如下命令即可:

代码语言:javascript复制
192:JavaVirtualMachines zzs$ jdk9
192:JavaVirtualMachines zzs$ java -version
java version "9.0.4"
Java(TM) SE Runtime Environment (build 9.0.4 11)
Java HotSpot(TM) 64-Bit Server VM (build 9.0.4 11, mixed mode)

此时,已经切换为Java 9了。当然,如果想切换回Java 8,则执行jdk8即可。

通过这种别名的形式,可以配置多个JDK环境,但遇到一些复杂的环境,这种简单的手动切换就显得力不从心了。我们需要更加便捷,细粒度的控制工具,这里推荐JEnv这款工具。

JEnv工具使用

如官网所说JEnv是一款让你忘记如何配置JAVA_HOME的神器,使用简单的命令就可以在不同Java版本之间进行切换。下面就来看看JEnv的安装和基本使用。

JEnv的安装

在Mac OS下可使用Homebrew安装JEnv:

代码语言:javascript复制
brew install jenv

等待上述命令安装成功后,还需在.bash_profile中进行一下配置。对应的配置内容,在安装时控制台也有打印,这里通过echo命令直接将内容添加到.bash_profile文件当中:

代码语言:javascript复制
$ echo 'export PATH="$HOME/.jenv/bin:$PATH"' >> ~/.bash_profile
$ echo 'eval "$(jenv init -)"' >> ~/.bash_profile

安装完成,执行命令查询一下版本信息:

代码语言:javascript复制
$ jenv versions
* system (set by /Users/zzs/.jenv/version)

从结果看出,只找到了系统默认的Java,*表示当前选择的版本。尽管安装了其他版本的Java,但没有自动发现。

还是以现在的Java 8和Java 9环境为例,我们通过jenv add命令将对应的Java路径添加到jenv中:

代码语言:javascript复制
$ jenv add /Library/Java/JavaVirtualMachines/jdk1.8.0_151.jdk/Contents/Home/
oracle64-1.8.0.151 added
1.8.0.151 added
1.8 added
 1.8.0.151 already present, skip installation

Java 9添加到jenv中:

代码语言:javascript复制
$ jenv add /Library/Java/JavaVirtualMachines/jdk-9.0.4.jdk/Contents/Home/
oracle64-9.0.4 added
9.0.4 added
9.0 added
 9.0.4 already present, skip installation

此时再运行查询版本信息命令:

代码语言:javascript复制
$ jenv versions
* system (set by /Users/zzs/.jenv/version)
  1.8
  1.8.0.151
  9.0
  9.0.4
  oracle64-1.8.0.151
  oracle64-9.0.4

可以看到Java 8和Java 9的JDK已经添加到jenv当中了。但很明显,也添加了一些无需的JDK版本,可以通过remove命令将其移除:

代码语言:javascript复制
$ jenv remove oracle64-1.8.0.151
JDK oracle64-1.8.0.151 removed

当不需要的版本移除之后,再次查看:

代码语言:javascript复制
$ jenv versions
* system (set by /Users/zzs/.jenv/version)
  1.8.0.151
  9.0.4

只留下了1.8.0.151和9.0.4这两个版本。

JEnv的使用

完成了上述安装,便可以通过命令来切换所使用的Java版本了:

代码语言:javascript复制
$ jenv local 1.8.0.151
/usr/local/Cellar/jenv/0.5.5_2/libexec/libexec/jenv-version-file-write: line 19: .java-version: Permission denied

在尝试使用时,并没有出现预期的效果,报权限问题,这里可通过doctor命令来查看一下原因:

代码语言:javascript复制
$ jenv doctor
[ERROR]	JAVA_HOME variable already set, scripts that use it directly could not use java version set by jenv
[OK]	Java binaries in path are jenv shims
[OK]	Jenv is correctly loaded

第一行提示JAVA_HOME变量已经配置了,于是编辑.bash_profile文件注释掉原来的JAVA_HOME变量,再次执行:

代码语言:javascript复制
$ jenv doctor
[OK]	No JAVA_HOME set
[OK]	Java binaries in path are jenv shims
[OK]	Jenv is correctly loaded

经过上述操作,发现在切换Java版本时“Permission denied”问题依旧出现,于是退出了命令行窗口,重新打开问题就消失了。

切换前版本:

代码语言:javascript复制
$ java -version
openjdk version "17.0.4.1" 2022-08-12
OpenJDK Runtime Environment Temurin-17.0.4.1 1 (build 17.0.4.1 1)
OpenJDK 64-Bit Server VM Temurin-17.0.4.1 1 (build 17.0.4.1 1, mixed mode, sharing)

执行切换,及切换后版本:

代码语言:javascript复制
$ jenv local 1.8
$ java -version
java version "1.8.0_151"
Java(TM) SE Runtime Environment (build 1.8.0_151-b12)
Java HotSpot(TM) 64-Bit Server VM (build 25.151-b12, mixed mode)

其中jenv local指定了某文件夹中local的Java版本,还可以通过jenv global设置一个默认的Java版本,运行jenv which java显示可执行的Java的完整路径。

设置全局默认版本:

代码语言:javascript复制
jenv global 1.8

显示可执行的Java的完整路径:

代码语言:javascript复制
$ jenv which java
/Users/zzs/.jenv/versions/1.8/bin/java

也可以在特定的文件夹下使用.java-version文件来设定Java的版本。

代码语言:javascript复制
$ cat .java-version
1.8

在执行过jenv local 1.8命令的目录下,可以看到生成了上述.java-version文件,并且文件内存储了Java的版本号。

比如,当在Project中使用Java 8时,仅仅需要把1.8作为内容保存在.java-version文件中,当进入该文件夹时,JEnv会自动地帮助我设定local的Java的版本。

PS:在上述操作的过程中,我们需要注意某些时刻未达到预期效果,可能是命令行窗口缓存的问题,可尝试关闭命令行窗口重新打开进行查看。

JEnv实践

这里通过Idea 2022创建一个基于Java 17的项目java17-learn:

代码语言:javascript复制
.
├── pom.xml
└── src
    ├── main
    │   ├── java
    │   │   └── com
    │   │       └── secbro2
    │   │           └── Main.java
    │   └── resources
    └── test
        └── java

创建完成之后,进入项目根目录,执行jenv命令将对应的目录的环境变量设置为Java 17:

代码语言:javascript复制
$ jenv local temurin64-17.0.4.1
$ java -version
openjdk version "17.0.4.1" 2022-08-12
OpenJDK Runtime Environment Temurin-17.0.4.1 1 (build 17.0.4.1 1)
OpenJDK 64-Bit Server VM Temurin-17.0.4.1 1 (build 17.0.4.1 1, mixed mode, sharing)

可以看到已经给指定目录设置了对应的Java版本。此时如果用IDEA打开项目,会看到项目的根目录下多一个.java-version的文件,打开该文件,内容为:

代码语言:javascript复制
temurin64-17.0.4.1

而新创建的这个项目与全局默认的Java 8的环境变量配置并不冲突,可以各自正常运行。至此,关于JEnv的使用介绍完毕。

小结

本篇文章针对多Java版本环境变量的管理问题,带大家实践和学习了两种环境变量管理方式:一种是比较粗放的,基于环境变量命令别名的形式,通过别名来切换当前的环境变量;另外一种是借助于三方工具JEnv来完成多Java版本的精确管理,可以细化到具体的文件目录。如果条件允许,推荐大家使用JEnv来进行管理。

0 人点赞