zoukankan      html  css  js  c++  java
  • Java Nashorn--Part 5

    Nashorn 的高级应用

    Nashorn 是一个复杂的编程环境,它被设计为一个强大的平台,用于部署应用程序,并与Java具有极大的互操作性。 让我们来看一些更高级的用于 JavaScript 到 Java 集成的用例,并通过在 Nashorn中 查看一些实现细节来掌握是如何实现的。

    从 Nashorn 中调用 Java

    由于每个 JavaScript 对象最后都被编译为 Java 类的实例,所以 Nashorn 与 Java 无缝集成也许并不奇怪,尽管在类型系统和语言特性方面有很大的区别。 然而,仍然需要有一些机制来充分利用这种集成。
    我们已经看到,我们可以直接从 Nashorn 访问 Java 类和方法,例如:

    $ jjs -Dkey=value
    jjs> print(java.lang.System.getProperty("key"));
    value
    

    让我们仔细看看是如何在 Nashorn 中实现这种支持的。

    JavaClass 和 JavaPackage

    从 Java 的角度来看, java.lang.System.getProperty("key")表达式有资格访问 java.lang.System 类下的静态方法 getProperty()。作为JavaScript 语法来说,这非常类似于属性链的访问方式,这个链从 java 这个标识符开始,来看下面的例子:

    jjs> print(java);
    [JavaPackage java]
    
    jjs> print(java.lang.System);
    [JavaClass java.lang.System]
    

    所以说java是Nashorn 中特定的一个对象用来访问 Java 系统里的包, java被定义为 JavaPackage 的 JavaScript 的类型,Java 类被定义为JavaClass 的JavaScript 的类型,任何顶层包可直接用作包导航对象,以及子包可以被分配为一个 JavaScript 对象。这提供了访问 Java 类的简明语法:

    jjs> var juc = java.util.concurrent;
    jjs> var chm = new juc.ConcurrentHashMap;
    

    除了导航包对象,还有另一个对象,称为 Java,其中有一些关于它的有用的方法。其中最重要的是Java type()方法,这允许用户查询 Java 类型系统,并获得java类。例如:

    jjs> var clz = Java.type("java.lang.System");
    jjs> print(clz);
    [JavaClass java.lang.System]
    

    如果这个类不在指定的 classpath 中 (例如,使用 jjs 选择 -cp 选项指定),则会抛出 ClassNotFoundException 异常(jjs 会处理为 RuntimeException 异常)。

    jjs> var klz = Java.type("Java.lang.Zystem");
    java.lang.RuntimeException: java.lang.ClassNotFoundException:
      Java.lang.Zystem
    

    在大多数情况下,在 JavaScript 下 JavaClass 类型就像 Java 中的对象一样使用(它们有一点点不同,基本上可以认为它们是Nashorn级别上的类的影像),举例,我们可以使用 JavaClass 从 Nashorn 中直接创建 Java 对象。

    jjs> var clz = Java.type("java.lang.Object");
    jjs> var obj = new clz;
    jjs> print(obj);
    java.lang.Object@73d4cc9e
    
    jjs> print(obj.hashCode());
    1943325854
    
    // Note that this syntax does not work
    jjs> var obj = clz.new;
    jjs> print(obj);
    undefined
    

    JavaScript 和 Java Lambda 表达式

    JavaScript 和 Java 之间的互操作性达到了非常深的层次。我们甚至可以使用JavaScript 函数作为 Java 接口的匿名实现(或作为lambda表达式)。 例如,让我们使用一个 JavaScript 函数作为 Callable 接口的一个实例,这个接口只有一个方法 call(),它不需要参数并返回 void。 在 Nashorn 中,我们可以使用 JavaScript 函数作为 lambda 表达式的实现:

    jjs> var clz = Java.type("java.util.concurrent.Callable");
    jjs> print(clz);
    [JavaClass java.util.concurrent.Callable]
    jjs> var obj = new clz(function () { print("Foo"); } );
    jjs> obj.call();
    Foo
    

    正如事实所示,基本上,在 Nashorn 中,JavaScript 函数和 Java lambda 表达式之间没有区别。 正如我们在 Java 中看到的,函数被自动转换为适当类型的对象。让我们看看如何使用 Java 的 ExecutorService 接口在线程池上执行 Nashorn 的 JavaScript 代码。

    jjs> var juc = java.util.concurrent;
    jjs> var exc = juc.Executors.newSingleThreadExecutor();
    jjs> var clbl = new juc.Callable(function (){
      java.lang.Thread.sleep(10000); return 1; });
    jjs> var fut = exc.submit(clbl);
    jjs> fut.isDone();
    false
    jjs> fut.isDone();
    true
    

    与等效的 Java 代码相比(即便使用 Java 8 中的 lambdas 表达式),样板代码的减少是相当惊人的。 然而,由于使用了 lambda 的方式,存在一些限制。 例如:

    jjs> var fut=exc.submit(function (){
    java.lang.Thread.sleep(10000); return 1;});
    java.lang.RuntimeException: java.lang.NoSuchMethodException: Can't 
    unambiguously select between fixed arity signatures
    [(java.lang.Runnable), (java.util.concurrent.Callable)] of the method
    java.util.concurrent.Executors.FinalizableDelegatedExecutorService↵
    .submit for argument types
    [jdk.nashorn.internal.objects.ScriptFunctionImpl]
    

    这里的问题是 ExecutorService 有一个重载的 submit() 方法。 一个方法参数为 Callable,另一个重载的方法参数为 Runnable。 不幸的是,JavaScript 函数适用于转换为两种类型(包括 lambda 表达式)。虽然在运行时可以选择其中一个,但不能在它们之间进行选择 ,因此产生了无法“明确选择”的错误消息。

  • 相关阅读:
    新能源汽车三大核心技术
    新能源汽车分类解读
    hadoop(四):配置参数
    HDP2.4安装(六):小结与回顾
    【Web】Nginx配置开机启动
    【Linux】ODBC安装
    【Erlang】源码安装
    【Linux】 Ncures库的介绍与安装
    【RabbitMQ】 RabbitMQ安装
    《MinDoc 接口文档在线管理系统》
  • 原文地址:https://www.cnblogs.com/IcanFixIt/p/6407008.html
Copyright © 2011-2022 走看看