[英]I'd like to use `jshell` with JDK 8
我最近试用了 JDK 11 jshell
,喜欢能够以交互方式测试核心 Java 中记录的内容。 不过,我的公司仍然与 Java 8 一起发布。 所以我想我会尝试看看我是否可以使用jshell
来处理 JDK 8。
编辑以尝试解释似乎不清楚的内容:我想使用 JShell,但将 Java 8 API 用于交互模式、脚本、完成、JavaDoc 等。
我对jshell
正在做的事情的心理模型是:它是用它提供的 JDK 构建的,因此它在它为其构建的 JRE 上运行,但我认为它从 JDK 集合中加载了 JavaDoc,自动完成jars,并针对另一个 JVM 实例运行交互式脚本,不一定是与它运行的版本相同的实例。 所以我想我可以用JAVA_HOME
环境变量控制第二部分。
在 Mac 上,我简单的第一次尝试是:
JAVA_HOME=/Library/Java/JavaVirtualMachines/jdk1.8.0_144.jdk/Contents/Home/ \
/Library/Java/JavaVirtualMachines/jdk-11.0.1.jdk/Contents/Home/bin/jshell
但对我来说这并不明显(我会换行):
jshell> System.getProperties()
$3 ==> {gopherProxySet=false, awt.toolkit=sun.lwawt.macosx.LWCToolkit,
java.specification.version=11, sun.cpu.isalist=, sun.jnu.encoding=UTF-8,
java.class.path=., java.vm.vendor=Oracle Corporation, sun.arch.data.model=64,
java.vendor.url=http://java.oracle.com/, user.timezone=, os.name=Mac OS X,
java.vm.specification.version=11, sun.java.launcher=SUN_STANDARD, user.country=US,
sun.boot.library.path=/Library/Java/JavaVirtualMachines/jdk-11.0.1.jdk/Contents/Home/lib,
sun.java.command=jdk.jshell.execution.RemoteExecutionControl 55110,
jdk.debug=release, sun.cpu.endian=little, user.home=/Users/lamblin,
user.language=en, java.specification.vendor=Oracle Corporation,
...
patch.level=unknown,
java.library.path=/Users/lamblin/Library/Java/Extensions:/Library/Java/Extensions:/Network/Library/Java/Extensions:/System/Library/Java/Extensions:/usr/lib/java:.,
java.vendor=Oracle Corporation, java.vm.info=mixed mode,
java.vm.version=11.0.1+13-LTS, sun.io.unicode.encoding=UnicodeBig,
java.class.version=55.0}
jshell> System.getenv("JAVA_HOME")
$4 ==> "/Library/Java/JavaVirtualMachines/jdk1.8.0_144.jdk/Contents/Home/"
jshell> exit
为什么? Stream.ofNullable(T)
是 Java 9 中的新内容,虽然我希望它告诉我它不在 Java 8 中,但它可以正常工作。
jshell> Stream.ofNullable(null).iterator().hasNext()
$3 ==> false
jshell> Stream.ofNullable("hi").iterator().hasNext()
$4 ==> true
jshell> Stream.ofNullable(
Signatures:
Stream<T> Stream<T>.<T>ofNullable(T t)
<press tab again to see documentation>
jshell> Stream.ofNullable(
Stream<T> Stream<T>.<T>ofNullable(T t)
Returns a sequential Stream containing a single element, if non-null, otherwise returns an
empty Stream .
Type Parameters:
T - the type of stream elements
Parameters:
t - the single element
Returns:
a stream with a single element if the specified element is non-null, otherwise an empty stream
<press tab again to see all possible completions; total possible completions: 544>
jshell> Stream.ofNullable(
也许这个jshell
不可能实现我想要的。 当我看到一个在 JDK 8 上的 NetBeans 中使用jshell
的视频时,我开始认为这是。该视频可能误报了实际完成的内容。
我确实发现更改类路径并不是我所追求的,因为这会阻止jshell
加载它需要的类。
所以不是: /Library/Java/JavaVirtualMachines/jdk-11.0.1.jdk/Contents/Home/bin/jshell --class-path /Library/Java/JavaVirtualMachines/jdk1.8.0_144.jdk/Contents/Home/jre/lib/*
可以正常启动jshell
,然后使用/env -class-path /Library/Java/JavaVirtualMachines/jdk1.8.0_144.jdk/Contents/Home/jre/lib/rt.jar
将JRE 8 jar之一添加到类路径/env -class-path /Library/Java/JavaVirtualMachines/jdk1.8.0_144.jdk/Contents/Home/jre/lib/rt.jar
。 这可以使用jre/lib
所有 jar 完成,但它不会卸载或替换已加载的内容,因此它最终不会为您提供仅限于运行 Java 8 片段的环境。
这不是在 Java 8 中使用jshell
。
JAVA_HOME=/Library/Java/JavaVirtualMachines/jdk1.8.0_144.jdk/Contents/Home/ \
/Library/Java/JavaVirtualMachines/jdk-11.0.1.jdk/Contents/Home/bin/jshell
为什么? 因为 JDK 工具实际上并不使用 $JAVA_HOME 来决定使用哪个 JVM。 而是由可执行文件(或包装脚本)根据其文件位置做出决定。
但是您可以通过运行jshell -version
来确认这一点。 (注:只有一个-
!!)
如果下载 OpenJDK 源码,可以看到jshell
大多是用 Java 实现的。 它有一个主要的方法。 因此,您可以使用 Java 8 编译器编译jshell
类。 但这很可能会失败1 ,因为jshell
知道模块,这意味着它依赖于 Java 9 中与模块相关的 API 更改。
但是……何必呢?
为什么?
Stream.ofNullable(T)
是 Java 9 中的新内容,虽然我希望它告诉我它不在 Java 8 中,但它可以正常工作。
Java 8 编译器会告诉你。 假设您设置了平台依赖项,IDE 也是如此。
您也许可以使用-C
选项让jshell
编译您的代码以针对 Java 8 API。
1 - 如果您努力解决这些预期的依赖项以及 jshell 类中的其他 Java 9+ 依赖项,那么您将成功地将其向后移植到 Java 8。名利等着您:-)
这就是我所做的 - 我有 Java 8 和 Java 17,我暂时转而使用 Java 17 进行 jshell 会话......
jshell() {
export JAVA_HOME=/opt/homebrew/Cellar/openjdk/17/libexec/openjdk.jdk/Contents/Home
/usr/bin/jshell "$@"
export JAVA_HOME=/Library/Java/JavaVirtualMachines/temurin-8.jdk/Contents/Home
}
我将上面的 bash 函数添加到我的~/.bash_profile
虽然这可能不是完美的方式,但至少我可以尝试一些在 Java 8 和 Java 17 中通用的东西......
~ java -version
openjdk version "1.8.0_302"
OpenJDK Runtime Environment (Temurin)(build 1.8.0_302-b08)
OpenJDK 64-Bit Server VM (Temurin)(build 25.302-b08, mixed mode)
~ jshell
| Welcome to JShell -- Version 17
| For an introduction type: /help intro
jshell> String s = "abc"
s ==> "abc"
jshell> s
s ==> "abc"
jshell> s.charAt(1)
$4 ==> 'b'
jshell> /exit
| Goodbye
~ java -version
openjdk version "1.8.0_302"
OpenJDK Runtime Environment (Temurin)(build 1.8.0_302-b08)
OpenJDK 64-Bit Server VM (Temurin)(build 25.302-b08, mixed mode)
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.