java8使用nashorn与javascript交互

  • 指定重載方法

实际上方法在 JavaScript 不过是个特性,为函数实例,因此,除了使用存在的方法之外,也可以使用 [] 来取得代表方法的函式:

var System = Java.type('java.lang.System');
System.out.println('Hello, World');    // Hello, World
System.out['println']('Hello, World'); // Hello, World

Java 中支援重載(Overload)方法,例如,System.outprintln 有多個重載版本,如果你想指定调用特定的重載版本,可以使用 [] 時指定参数列。例如:

var System = Java.type('java.lang.System');
System.out['println'](3.14);          // 3.14
System.out['println(double)'](3.14);  // 3.14
System.out['println(int)'](3.14);     // 3

實例方法也可以使用這種方式:

var PrintStream = Java.type('java.io.PrintStream');
var file = new PrintStream('test.txt');
file['println(int)'](3.14);  // test.txt 中只存 3 這個數值
file.close();
  • JavaImporter、importPackage、 importClass

Java 中有 import 語法,在 Nashorn 中可以用 Java.type 來模擬 import 特定類別的作用:

var PrintStream = Java.type('java.io.PrintStream');

如果想同時將數個類別或套件放在某個名稱空間下,可以使用 JavaImport。例如:

var commons = new JavaImporter(java.util, java.lang);
with(commons) {
    var lt = Arrays.asList(1, 2, 3);
    System.out.println(lt);   // [1, 2, 3]
}

Nashorn 內建了 mozilla_compat.js,可以使用 load 載入後,使用 imoprtPackageimportClass,前者相當於在 Java 中 import 時在類別的部份使用 *,後者用來 import 某個類別。例如:

load('nashorn:mozilla_compat.js');

importClass(java.lang.System);
importPackage(java.util);

System.out.println('Hello, World'); // Hello, World

var lt = Arrays.asList(1, 2, 3);
System.out.println(lt);             // [1, 2, 3]
  • 實作介面

如果想要實作介面,可以使用 Java.extend。例如:

load('nashorn:mozilla_compat.js');

importPackage(java.lang);

var RunDemo = Java.extend(Runnable, {
    run: function() {
        [1, 2, 3].forEach(print);
    }
});

var th = new Thread(new RunDemo());
th.start();
th.join();

不使用 Java.extend,也可以用類似 Java 的匿名類別實作語法:

load('nashorn:mozilla_compat.js');

importPackage(java.lang);

var r = new Runnable {
    run: function() {
        [1, 2, 3].forEach(print);
    }
};

var th = new Thread(r);
th.start();
th.join();

如果方法的參數型態是個只具備單一抽象方法的介面,可以直接使用函式實作。例如:

var IntStream = Java.type("java.util.stream.IntStream");
var sum = IntStream.of(1, 2, 3, 4, 5, 6)
                   .filter(function(elem) {
                       return elem > 2;
                   })
                   .sum();
print(sum); // 18
  • 繼承類別

如果是要繼承抽象類別並實作抽象方法,方式與實作介面是類似的:

load('nashorn:mozilla_compat.js');

importPackage(java.util);

var r = new TimerTask {
    run: function() {
        [1, 2, 3].forEach(print);
    }
};

r.run();

var DemoTask = Java.extend(TimerTask, {
    run: function() {
        [1, 2, 3].forEach(print);
    }
});

new DemoTask().run();

然而,如果是繼承非抽象類別,就必須使用 java.extend

load('nashorn:mozilla_compat.js');

importPackage(java.lang);

var DemoThread1 = Java.extend(Thread, {
    run: function() {
        [1, 2, 3].forEach(print);
    }
});

new DemoThread1().run();

var DemoThread2 = Java.extend(Thread);

(new DemoThread2 {
    run: function() {
        [1, 2, 3].forEach(print);
    }
}).run();

实际上,Java.extend 每次调用后,都會建立一個子類別的型態物件,因此,以下两种写法有些不同:

load('nashorn:mozilla_compat.js');

importPackage(java.lang);

var DemoThread1 = Java.extend(Thread, {
    run: function() {
        [1, 2, 3].forEach(print);
    }
});

var DemoThread2 = Java.extend(Thread, {
    run: function() {
        [1, 2, 3].forEach(print);
    }
});

System.out.println(new DemoThread1().getClass() === new DemoThread2().getClass()); // false

var DemoThread = Java.extend(Thread);

var th1 = new DemoThread {
    run: function() {
        [1, 2, 3].forEach(print);
    }
};

var th2 = new DemoThread {
    run: function() {
        [1, 2, 3].forEach(print);
    }
};

System.out.println(th1.getClass() === th2.getClass());  // true
  • 调用父类方法

如果想调用父类实例化並重写某方法,可以如下:

load('nashorn:mozilla_compat.js');

importPackage(java.lang);

var Th = Java.extend(Thread);

var r = function() { 
    [1, 2, 3].forEach(function(elem) {
        print(elem);
    });
};

var th1 = new Th(r) {
    run : function() {
        Java.super(th1).run();
        [4, 5, 6].forEach(function(elem) {
            print(elem);
        });
    }
};

th1.start();
th1.join();

上面的范例也示范了 Java.super 的使用,它可用来调用父类别的方法。

  • 从 Java 中执行 JavaScript

要在 Java 中使用 Nashorn 執行 JavaScript 的話,可以如下取得 Nashorn 引擎:

ScriptEngineManager factory = new ScriptEngineManager();
ScriptEngine engine = factory.getEngineByName("nashorn");
engine.eval("print('Hello, World!')");
engine.eval(new FileReader("C:\\workspace\\hello.js"));
  • 参考:
  1. https://openhome.cc/Gossip/CodeData/JDK8/Nashorn-1.html
  2. https://openhome.cc/Gossip/CodeData/JDK8/Nashorn-2.html
  3. https://openhome.cc/Gossip/CodeData/JDK8/Nashorn-3.html
  • 1
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Nashorn是JDK 1.8中提供的一个JavaScript引擎,它可以让Java程序直接运行JavaScript代码。下面是使用Nashorn JavaScript引擎的详细步骤: 1. 导入Nashorn库 在Java代码中导入Nashorn库。在Java 8中,Nashorn库已经默认包含在JDK中,因此不需要额外下载和导入。 2. 创建一个Nashorn引擎 通过调用ScriptEngineManager类的getEngineByName()方法并传入“nashorn”作为参数,可以创建一个Nashorn引擎。例如: ``` ScriptEngine engine = new ScriptEngineManager().getEngineByName("nashorn"); ``` 3. 执行JavaScript代码 可以通过以下方式执行JavaScript代码: - 从一个文件中读取JavaScript代码并执行: ``` engine.eval(new FileReader("file.js")); ``` - 直接执行字符串中的JavaScript代码: ``` engine.eval("print('Hello, world!')"); ``` 4. 在JavaJavaScript之间传递变量 可以通过Bindings对象在JavaJavaScript之间传递变量。Bindings对象充当了一个映射表,可以将Java变量映射到JavaScript变量,也可以将JavaScript变量映射到Java变量。例如: ``` Bindings bindings = engine.createBindings(); bindings.put("message", "Hello, world!"); engine.eval("print(message)", bindings); ``` 在这个例子中,创建了一个Bindings对象,并将一个名为“message”的字符串变量绑定到它上面。然后,将这个Bindings对象传递给eval()方法,以便在JavaScript代码中使用该变量。 5. 调用JavaScript函数 可以使用ScriptEngine的get()方法获得一个函数对象,并使用函数对象的call()方法调用JavaScript函数。例如: ``` engine.eval("function add(a, b) { return a + b; }"); Invocable invocable = (Invocable) engine; Object result = invocable.invokeFunction("add", 1, 2); System.out.println(result); // 输出3 ``` 在这个例子中,通过eval()方法定义了一个名为“add”的JavaScript函数。然后,通过调用invokeFunction()方法调用了这个函数,并传递了两个参数。最后,将函数返回的结果打印到控制台上。 希望这些步骤能帮助你了解如何使用Nashorn JavaScript引擎。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值