我正在尝试从 Java 应用程序中嵌入和评估 ruby 代码。我需要能够使用随 rvm 安装的 jruby 环境,而不是将 jruby-complete.jar 放在我的类路径中。我可以执行基本的内核代码,但我遇到了需要标准库(fileutils、tmpdir 等)的问题。
我在下面创建了使用通过 RVM 安装的 JRuby 的测试文件,如果您有本地 rvm + jruby 安装(将 JRUBY_VERSION 更改为安装的版本),任何人都应该能够编译+运行它。我意识到我引用的 jruby.jar 与 jruby-complete.jar 不同,但我希望有一种方法可以在不下载外部 jar 的情况下加载标准库。
import java.io.File;
import java.lang.reflect.Method;
import java.net.URL;
import java.net.URLClassLoader;
import java.util.logging.Logger;
import javax.script.ScriptEngine;
import javax.script.ScriptEngineManager;
public class Test {
private final static Logger LOG = Logger.getAnonymousLogger();
private final static String JRUBY_VERSION = "jruby-1.6.7";
public static void main(String[] args) throws Throwable {
final String rvmPath = System.getenv("HOME") + "/.rvm/rubies/";
addFileToClasspath(rvmPath + JRUBY_VERSION + "/lib/jruby.jar");
final ScriptEngine rubyEngine = new ScriptEngineManager().getEngineByName("jruby");
rubyEval(rubyEngine, "puts 'hello world!'"); // works
rubyEval(rubyEngine, "require 'tempfile'"); // fails to load 'tmpdir'
rubyEval(rubyEngine, "require 'fileutils'"); // fails to load 'fileutils'
}
private static void rubyEval(ScriptEngine rubyEngine, final String code) {
try {
rubyEngine.eval(code);
} catch (final Throwable e) { LOG.throwing(Test.class.getName(), "rubyEval", e); };
}
public static void addFileToClasspath(final String path) throws Throwable {
final File file = new File(path);
final URLClassLoader sysloader = (URLClassLoader) ClassLoader.getSystemClassLoader();
final Class<?> sysclass = URLClassLoader.class;
final Method method = sysclass.getDeclaredMethod("addURL", new Class<?>[] {URL.class});
method.setAccessible(true);
method.invoke(sysloader, new Object[] {file.toURI().toURL()});
}
}