Js.cast()
是一种欺骗的方式,并且做一些 Java 语言不允许但实际上可能是合法的事情。忽略“它实际上是如何工作的”,这个想法是你现在可以解决 Java 会抱怨的问题,即使它被证明是合法的。
一个示例可能是您使用 ajava.lang.Double
并double
希望将其视为 a JsNumber
,因此您可以对其调用 toPrecision(2) 。由于java.lang.Double
是 final 的,强制转换为不相关的类型是不合法的,但 Java 不知道在 GWT 中, Double 真的只是一个 js Number
。因此,您可以改为使用Js.cast()
. 编译器将在其中插入运行时类型检查,在运行时验证您的号码实际上是 JS Number 实例。
另一个示例可能是尝试扩展 elemental2 提供的某些本机类型,以实现缺失功能的解决方法,或执行特定于浏览器的操作。您的新类可能不会扩展现有类 - 从 JS 的角度来看,这没关系,您只是在描述您知道将在运行时存在的 API。因此,我们需要避免“这种转换是否有意义?”的 Java 语言检查,而只是告诉编译器尝试一下。
另一方面,你可以用Js.uncheckedCast()
. 这用于您甚至要求运行时跳过检查并假装它会工作的情况。这可以让你做一些奇怪的事情,比如把字符串当作数组来对待,或者解决跨框架问题。不会发出运行时检查,因此如果缺少方法/属性,您可能只会收到 TypeError,而不是正确的 ClassCastException。
DocumentRange
在 elemental2-dom 1.0.0-RC1 中,有一个类叫做一个“接口”(在 JS-land 中意味着它只是一种类型的描述,而不是你可以进行类型检查的东西)。https://www.w3.org/TR/DOM-Level-2-Traversal-Range/ranges.html#Level2-DocumentRange-method-createRange
这个错误是从closure-compiler继承的,它声称它有一个构造函数:https ://github.com/google/closure-compiler/blob/6a418aa/externs/browser/w3c_range.js#L241-L251
解决方法是让闭包编译器将此称为接口,并制作 elemental2 的新版本,以便您可以使用它。
您可以在此处进行两种解决方法。第一种是作弊Js.uncheckedCast(DomGlobal.document)
并说“是的,我知道Document
不是instanceof DocumentRange
,但那是因为没有像这样的类DocumentRange
,所以就假装它有效,这样我就可以调用createRange()
它”。这就是您已经在做的事情 - 它隐藏了存在错误的事实,但最终它可以工作。
“正确”的答案是声明你自己的DocumentRange
,然后Js.cast()
改为这样做。这仍然很糟糕——你必须保留你的新界面,直到闭包得到修复,然后 elemental2 被释放,然后你必须记住清理它。
在这种情况下,我建议对 GWT 撒谎并使用Js.uncheckedCast()
- 这里只有一个方法,而且不太可能以有意义的方式改变。