How to convert Rhino-JavaScript arrays to Java-Arrays
Asked Answered
B

5

18

I have the following:

ScriptEngineManager mgr = new ScriptEngineManager();
ScriptEngine jsEngine = mgr.getEngineByName("JavaScript");
jsEngine.eval("function getArray() {return [1,2,3,4,5];};");
Object result = jsEngine.eval("getArray();");

How can i convert the result object which is of type sun.org.mozilla.javascript.internal.NativeArray to a corresponding java array? Can somone show me a working code sample where this is done? It should work for String and Integer arrays. Plus, it would be great to know where to look for other data type conversions between the rhino engine and java.

Btw, i know this page but i'm really looking for a working code sample.

Berk answered 16/9, 2009 at 14:36 Comment(0)
E
22
NativeArray arr = (NativeArray) result;
Object [] array = new Object[(int) arr.getLength()];
for (Object o : arr.getIds()) {
    int index = (Integer) o;
    array[index] = arr.get(index, null);
}
Equanimous answered 16/9, 2009 at 14:49 Comment(2)
thanks for helping a stupid guy ;-) i wish all answers would be that precise,short and working!Berk
Wonder why though you can't cast the entire getIds array to Integer[]. I'm getting a ClassCastException. Oh wellTorrid
D
9

I'm not sure if it was the case when this question was first asked, but NativeArray implements the java.util.List interface. A simple way to convert to a real Java array is therefore:

Object[] array = ((List<?>) result).toArray();
Decoder answered 5/12, 2013 at 16:9 Comment(3)
jdk1.6.0_45 says sun.org.mozilla.javascript.internal.NativeArray cannot be cast to java.util.ListSalpa
@Salpa NativeArray definitely implements List in more recent versions of Rhino, but I'm not sure when that was first the case. See here.Decoder
By the way, Nashorn (the JavaScript engine included with Java 8) unfortunately returns objects of a class (ScriptObjectMirror) that implements Map instead of List. If you want your code to work under both Java 7 and 8+, you'll have to do an instanceof check.Lust
R
8

If the Javascript is under your control, you could do the transformation there, as per this document. So to adapt your example, something like:

ScriptEngineManager mgr = new ScriptEngineManager();
ScriptEngine jsEngine = mgr.getEngineByName("JavaScript");
jsEngine.eval("function getArray() {return [1,2,3,4,5];};");
String convertFuncSrc =
     "function convertArray(type, arr) {"
   + "  var jArr = java.lang.reflect.Array.newInstance(type, arr.length);"
   + "  for (var i = 0; i < arr.length; i++) { "
   + "    jArr[i] = arr[i];"
   + "  }"
   + "  return jArr;"
   + "};";
jsEngine.eval(convertFuncSrc);
Object result = jsEngine.eval("convertArray(java.lang.Integer.TYPE, getArray());");
int[] javaArray = (int[])result;

Although, if you can't change the Javascript this approach won't work, and you [i]will[/i] have an instance of sun.org.mozilla.javascript.internal.NativeArray as your result variable. At which point I think you just need to cast and deal with it directly, using whatever public methods it exposes; it's probably not pretty but I don't see any other options. In particular I think the only thing you can guarantee at the nice Rhino level is that it will be an instance of Scriptable (and probably ScriptableObject), which doesn't help you use it as an Array.

Kevin's answer looks like a good way to go here (and is similar to what I was just about to edit in! :-))

Reimpression answered 16/9, 2009 at 14:47 Comment(0)
R
1

In my case I wanted to produce a Java array within the script. (This use case also matches the question.)

Following Creating Java Arrays, I came up with

var javaArray = java.lang.reflect.Array.newInstance(java.lang.Integer, jsArray.length);
for (var i = 0; i < javaArray.length; i++) {
  javaArray[i] = new java.lang.Integer(jsArray[i]);
}
Ruttger answered 8/10, 2020 at 19:15 Comment(0)
L
0

General solution using JASON as data intermediate:

Object result = jsEngine.eval("JSON.stringify(getArray())");    
JsonReader jsonReader = Json.createReader(new StringReader(result.toString()));
JsonArray jsonArray = jsonReader.readArray();
jsonReader.close();
int[] numbers = new int[jsonArray.size()];
int index = 0;
for(JsonValue value : jsonArray){
    numbers[index++] = Integer.parseInt(value.toString());
}
Laxation answered 28/8, 2017 at 15:42 Comment(0)

© 2022 - 2024 — McMap. All rights reserved.