一些小的 javascript 片段可以很方便的直接使用 ScriptEngine.eval() 直接导入,但是如何使用 require 方法导入使用 npm 下载的 javascript 模块呢?
答案是使用 jvm-npm 这个项目。
首先,创建一个目录放置所有 javascript 文件,比如:D:\\path\\to\\js_modules\\
从 jvm-npm 下载 jvm-npm.js,放到上述目录中
使用 npm i <js_module> - g 下载你需要引用的 npm 模块,注意看最后的提示,即模块下载目录
到模块下载目录,通常为 <User_Dir>\npm\node_modules\<js_module>,里面一般有个 dist 目录,从里面找到完整的模块 js 文件,复制到上述目录
见以下 Kotlin 示例代码:(注意把 js_module 换成你需要加载的模块名)
val JS_ROOT = “D:\\path\\to\\js_modules”
val JSE = ScriptEngineManager().getEngineByName(“nashorn”).apply {
System.setProperty(“user.dir”, JS_ROOT) // 设定 jvm-npm 的当前目录
eval(Util.readLocalUtfText(“$JS_ROOT\\jvm-npm.js”)) // 在 ScriptEngine 中加载 jvm-npm.js
println(eval(“typeof require”)) // 验证 require 方法已正确引入
eval(“var mylib = require(‘js_module’)”) // 这时候,就可以用 require() 方法来引入当前目录中的 js 模块了
println(eval(“typeof mylib”)) // 验证 js 模块已正确加载
}
…
val bar = JSE.eval(“mylib.foo()”) // js 代码中可以使用加载的 js 模块了
注意:
jvm-npm 有个 bug,如果你的 js 文件大于 64K,则会导致文件截断。这种情况下你需要编写一个能一次读取完整 File 或 InputStream 的 java 方法,替换掉下面这句:
return new Scanner(input).useDelimiter(‘\\A’).next()