MVP For GWT 系列资料十二:彻底了解 GWT Part 2:JavaScript 的 overlay type

源文出自:https://txt.appspot.com/pt2club.blogspot.com/2010/02/gwt-part-2javascript-overlay-type.html

彻底了解 GWT Part 2:JavaScript 的 overlay type

原文:http://googlewebtoolkit.blogspot.com/2008/08/getting-to-really-know-gwt-part-2.html

技术校正、审阅:tkcn

假设你已经在 GWT module 当中,愉快地使用 JSNI 来呼叫某些手写的 JavaScript。一切运作正常,但是 JSNI 只能在独立的 method 下运作。某些整合性状况需要你彻底地把 JavaScript 跟 Java 的 object 绑在一起——写 DOM 跟 JSON 就是两个好例子——所以我们十分需要可以从 Java 程式码直接与 JavaScript object 互动的方法。换句话说,我们想要 JavaScript 的 object 看起来就像我们写的 Java object。

GWT 1.5 引入了 JavaScript overlay type,这让 GWT 程式整合各种 JavaScript object 变得容易许多。这个技术有很多好处,像是让你能用 Java IDE 的 code completion 跟 refactoring 功能,即使你写的是 untype 的 JavaScript object。

范例:简单、有效率的 JSON
用一个范例来了解 overlay type 是最简单的方法。假设我们要存取一组「customer」数据,底层是用 JSON object。在 JavaScript 中的资料结构可能像这样:

void jsonData = [

  { "FirstName" : "Ps", "LastName" : "Monkey" },

  { "FirstName" : "痞子", "LastName" : "" },

  { "FirstName" : "Pt2", "LastName" : "Club" },

  { "FirstName" : "STO", "LastName" : "Orz" },

];

 

要把一个 Java type 加到上述的资料结构,要从建立一个 JavaScriptObject 的 subclass 开始,这在 GWT 表示是一个 JavaScript 的 object。接著增加一些 getter。

// An overlay type

class Customer extends JavaScriptObject {

  // Overlay types always have protected, zero-arg ctors

  protected Customer() { }    

 

  // Typically, methods on overlay types are JSNI

  public final native String getFirstName() /*-{ return this.FirstName; }-*/

  public final native String getLastName()  /*-{ return this.LastName;  }-*/

 

  // Note, though, that methods aren't required to be JSNI

  public final String getFullName() {

    return getFirstName() + " " + getLastName(); 

  }

}

 

如此一来,GWT 就会了解所有 Customer 的 instance 实际上是来自 GWT module 以外的 JavaScript object。这包含了很多意义。举例来说,看到 getFirstName() 跟 getLastName() 里头的 this reference。它实质上是代表一个 JavaScript object,所以你操作这个 this 就像在 JavaScript 里头一样。在这个例子中,我们可以直接存取 JSON 中那些我们已知的 field:this.FirstName 跟 this.LastName。

 

那么,你要如何才能真正得到一个被包装成 Java type 的 JavaScript object 呢?你不能用 new Customer() 来建构它,因为重点是把一个既有的 JavaScript object 包装成 Java type。因此,我们必须使用 JSNI 来得到这样一个 object:

class MyModuleEntryPoint implements EntryPoint {

  public void onModuleLoad() {

    Customer c = getFirstCustomer();

    // Yay! Now I have a JS object that appears to be a Customer

    Window.alert("Hello, " + c.getFirstName());

  }

 

  // Use JSNI to grab the JSON object we care about

  // The JSON object gets its Java type implicitly 

  // based on the method's return type

  private native Customer getFirstCustomer() /*-{

    // Get a reference to the first customer in the JSON array from earlier

    return $wnd.jsonData[0];

  }-*/;

}

 

现在来搞清楚我们做了啥。我们拿到了一个 plain old JSON object(译注:源自于 POJO)并且建立一个看起来很正常的 Java type,让 GWT 程式码中能够使用它。于是你就有了 code completion、refactoring、compile 阶段的检查——这些写 Java 时所拥有的好处。然而,你还是可以灵活地操作任何 JavaScript object,这使得存取 JSON service(使用 RequestBuilder)变得很轻而易举。

 

为一些 compiler 强者岔题一下。overlay type 另一个美妙的事情是你可以增加 Java 的 type,但是却不用影响底层的 JavaScript object。注意到上面例子中,我们加入的 getFullName() 这个 method。它是纯粹的 Java 程式码(并不存在于底层的 JavaScript object),但却是依照底层 JavaScript object 所写的。也就是说,处理同一个 JavaScript object,以 Java 的角度会比用 JavaScript 功能丰富得多;而且不用动到底层的 JavaScript object——无论是 instance 或是 prototype。

 

(接续上一段的题外话)在 overlay type 增加 method 这个很酷的怪招是可行的,因为 overlay type 的设计规则是不允许 polymorphic 呼叫,所有的 method 必须是 final 且/或 private。因此,compiler 是静态地解读每一个 overlay type 的 method,所以不需要在 runtime 的时候动态 dispatch。这是为什么我们不用拘泥在 object 的 function pointer;compiler 可以直接对 method 呼叫,就好像是 global function、独立于 object 之外。很明显的,直接呼叫 function 会比间接快得多。更棒的是,因为呼叫 overlay type 的 method 是静态解读担庑┒。

 

范例:lightweight collection

 

我们在上面的例子当中掩盖了某些事情。getFirstCustomer() 这个 method 是非常不切实际的。你一定会希望存取全部的 customer 阵列。所以,我们需要一个 overlay type 来表示这个 JavaScript 阵列。幸运的是,这很简单:

//泛型在 overlay type 裡頭也運作正常!

class JsArray<E extends JavaScriptObject> extends JavaScriptObject {

  protected JsArray() { }

  public final native int length() /*-{ return this.length; }-*/;

  public final native E get(int i) /*-{ return this[i];     }-*/;

}

 

现在我们可以写出更有趣的程式了:

class MyModuleEntryPoint implements EntryPoint {

  public void onModuleLoad() {

    JsArray<Customer> cs = getCustomers();

    for (int i = 0, n = cs.length(); i < n; ++i) {

      Window.alert("Hello, " + cs.get(i).getFullName());

    }

  }

 

  // Return the whole JSON array, as is 

private final native JsArray<Customer> getCustomers() /*-{

    return $wnd.jsonData;

  }-*/;

}

 

这是一个很干净的程式码,尤其是以建立灵活配置的角度来看。正如上头提到的,compiler 可以作一些十分 fancy 的事情,让它相当有效率。看一下 onModuleLoad() 这个 method 在没有 obfuscate 的 compile 结果:

function $onModuleLoad(){

  var cs, i, n;

  cs = $wnd.jsonData;

  for (i = 0, n = cs.length; i < n; ++i) {

    $wnd.alert('Hello, ' + (cs[i].FirstName + ' ' + cs[i].LastName));

  }

}

 

这个最佳化真的是 xx 的好。即使是 getFullName() 这个 method 的 overhead 也没了。事实上, 所有 Java method 的呼叫动作都不见了。当我们说:「GWT 给你可负担的 abstraction」,这就是其中之一。不仅 inline 的程式码执行速度明显变快、我们不再需要定义 function 的内容、也因而得以将 script 简短化(虽然持平而论,inline 的方式也很容易让 script 量变多,所以我们小心地在速度与程式码大小之间取得平衡)。现在回顾上头原始的 Java 程式码是十分有趣的,而试著推导 compiler 最佳化的步骤就展示到这边。 不过,我们还是忍不住要 show 一下对应、obfuscate 过的程式码: 

function B(){var a,b,c;a=$wnd.jsonData;for(b=0,c=a.length;b<c;++b){  $wnd.alert(l+(a[b].FirstName+m+a[b].LastName))}}

 

注意在这个版本当中,唯一没有 obfuscate 的是 JavaScript 当中的识别字,例如 FirstName、 LastName、jsonData 等。这是为什么即使 GWT 努力让大量 JavaScript 交互沟通的操作变得容易,但我们还是努力说服别人尽量用纯 Java 来写程式、而不是混著 JavaScript 写。希望你听到我们讲这些之后,你会明白我们不是要打击 JavaScript——只是我们不能对它们最佳化,这会让我们很沮丧。

 

摻在一起作撒尿牛丸 

overlay type GWT 1.5 的重要特性。个技术让直接与 JavaScript library 互相沟通变得相当容易。希望在读完这篇文章之后,你可以想像如何以一组 Java type 直接导入任何 JavaScript library GWT 里头,进而使用 Java IDE 来进行高效率的开发跟 debug,却不会因为任何类型的 GWT overhead 而影响程式码大小或执行速度。同时,overlay type 作为一个强大的 abstraction 工具,提供更优雅的低阶 API,例如新的DOM package

你可能感兴趣的:(JavaScript,C++,c,json,gwt)