Кроме того, Rhino позволяет получать и изменять значения статических полей Java-классов и полей экземпляров Java-объектов из программы на языке JavaScript. В классах на языке Java часто не определяют общедоступные поля, отдавая предпочтение методам доступа. Если в Java-классе определены методы доступа, Rhino обеспечивает доступ к ним, как к свойствам объекта на языке JavaScript:
//
Прочитать значение статического поля Java-класса
var stdout = java.lang.System.out:
// Rhino отображает методы доступа в отдельные свойства JavaScript
f.name // => "/tmp/test": вызовет f.getName
f.directory // => false: вызовет f.isDirectory
В языке Java имеется возможность создавать перегруженные версии методов, имеющие одинаковые имена, но разные сигнатуры. Обычно интерпретатор Rhino способен определить, какую версию метода следует вызвать, опираясь на типы аргументов, которые передаются программой на языке JavaScript. Однако иногда бывает необходимо явно идентифицировать метод по имени и сигнатуре:
// Предположим, что Java-объект о имеет метод f, который принимает целое
// или вещественное число. В JavaScript необходимо будет явно указать сигнатуру:
о[’f(int)'](3); // Вызвать метод, принимающий целое число
о['f(float)'](Math.PI); // Вызвать метод, принимающий вещественное число
Для итераций по методам, полям и свойствам Java-классов можно использовать цикл
for/in
:
importClass(java.lang.System);
for(var m in System) print(m); // Выведет статические члены java.lang.System
for(m in f) print(m); // Выведет члены экземпляра java.io.File
// Обратите внимание, что таким способом нельзя перечислить классы в пакете
for (с in java.lang) print(c): // Этот прием не сработает
Rhino позволяет программам на языке JavaScript получать и изменять значения элементов Java-массивов, как если бы они были JavaScript-массивами. Конечно, Java-массивы отличаются от JavaScript-массивов: они имеют фиксированную длину, их элементы имеют определенный тип, и они не имеют JavaScript-методов, таких как
slice
. В JavaScript не существует синтаксических конструкций, которые могли бы использоваться интерпретатором Rhino для создания Java-массивов в программах на языке JavaScript, поэтому для этой цели необходимо использовать класс java.lang.reflect Array:
// Создать массив из 10 строк и массив из 128 байтов
var words = java.lang.reflect.Array.newlnstance(java.lang.String, 10);
var bytes = java.lang.reflect.Array.newlnstance(java.lang.Byte.TYPE, 128);
//
После создания с массивами можно работать как с JavaScript-массивами:
for(var і = 0; і < bytes.length; i++) bytes[i] = i;
Программирование на языке Java часто связано с реализацией интерфейсов. Чаще всего с этой необходимостью приходится сталкиваться при разработке графических интерфейсов, когда каждый обработчик события должен реализовать интерфейс приемника событий. Следующие примеры демонстрируют, как это сделать:
// Интерфейсы: Реализация интерфейсов выглядит следующим образом:
var handler = new java.awt.event.FocusListener({
focusGained: function(e) { printfgot focus"); },
focusLost: function(e) { print("lost focus"): }
}):
// Аналогично выполняется расширение абстрактных классов
Когда Java-метод возбуждает исключение, интерпретатор Rhino продолжает его распространение как JavaScript-исключения. Получить оригинальный Java-объект java.lang.Exception можно через свойство javaException JavaScript-объекта Error: