在 Java 中访问 JavaScript 对象的字段

Access fields of a JavaScript object in Java

本文关键字:对象 字段 JavaScript 访问 Java      更新时间:2023-09-26

我正在编写一个Scala应用程序(应该使用Spark在Hadoop上运行),我的用户将执行他们上传的JavaScript代码片段,我想为这些JavaScript用户提供对用Scala编写的某些辅助函数的访问(如"进行HTTP调用"等)。所以我所做的是编写一个大的JavaScriptHelpers对象,然后使用

engine = scriptEngineManager.getEngineByName("JavaScript")
engine.put("jql", JavaScriptHelpers)

因此用户可以在 JavaScript 中说jql.httpPost(...)。使这成为可能的 Scala 代码如下所示:

def httpPost(where: String, params: Object): Try[String] = {
  params match {
    // JavaScript string becomes Java's String:
    case body: String =>
      // ...
    // JavaScript object becomes Java's ScriptableObject
    case obj: ScriptableObject =>
      val params = convertToMap(obj)
      // ...
  }
}
protected def convertToMap(obj: ScriptableObject): Map[String, String] = {
  (for (key <- obj.getIds().toList) yield {
    (key.toString, obj.get(key) match {
      case s: String =>
        s
      case d: java.lang.Double if d.toString.endsWith(".0") =>
        d.toInt.toString
      case other => other.toString
    })
  }).toMap
}

我发现访问存储在 JavaScript 对象中的信息的唯一方法是将它们视为sun.org.mozilla.javascript.ScriptableObject的实例。现在这在我的本地 OpenJDK 安装上就像一个魅力

java version "1.7.0_75"
OpenJDK Runtime Environment (fedora-2.5.4.2.fc20-x86_64 u75-b13)
OpenJDK 64-Bit Server VM (build 24.75-b04, mixed mode)

但是当我在正在运行的 Hadoop 集群上运行相同的代码时

java version "1.7.0_67"
Java(TM) SE Runtime Environment (build 1.7.0_67-b01)
Java HotSpot(TM) 64-Bit Server VM (build 24.65-b04, mixed mode)

然后我得到:

java.lang.NoClassDefFoundError: sun/org/mozilla/javascript/ScriptableObject
    sun.org.mozilla.javascript.internal.JavaMembers.discoverAccessibleMethods(JavaMembers.java:383)
    sun.org.mozilla.javascript.internal.JavaMembers.discoverAccessibleMethods(JavaMembers.java:335)
    sun.org.mozilla.javascript.internal.JavaMembers.reflect(JavaMembers.java:455)
    sun.org.mozilla.javascript.internal.JavaMembers.<init>(JavaMembers.java:76)
    sun.org.mozilla.javascript.internal.JavaMembers.lookupClass(JavaMembers.java:847)
    sun.org.mozilla.javascript.internal.NativeJavaObject.initMembers(NativeJavaObject.java:88)
    sun.org.mozilla.javascript.internal.NativeJavaObject.<init>(NativeJavaObject.java:78)
    sun.org.mozilla.javascript.internal.NativeJavaObject.<init>(NativeJavaObject.java:68)
    ...

看看甲骨文与JDK 7捆绑在一起的Rhino版本,可以从 http://www.oracle.com/technetwork/opensource/jdk7-source-1634015.html 下载,似乎所有sun.org.mozilla.javascript.*类都已移至sun.org.mozilla.javascript.internal.*

现在我该如何处理这种情况?在Java中,是否有任何独立于Rhino的方式来访问JavaScript对象的字段?或者,如何强制 Oracle JVM 在本地环境中使用 ...javascript.ScriptableObject 时使用该...javascript.internal.ScriptableObject

任何帮助非常感谢。

您可以改用函数重载。

// `params` matches a JavaScript string
def httpPost(where: String, params: String): Try[String] = {
  // ...
}
// `params` matches a JavaScript object
def httpPost(where: String, params: java.util.Map[_, _]): Try[String] = {
  // ...
}

这个解决方案适用于我的环境(Oracle JDK 8和OpenJDK 1.7)。