2

我将 GWT 2.9 与 elemental2-1.0.0-RC1 一起使用。

以下代码ClassCastException在运行时抛出 a:

DocumentRange documentRange = Js.cast(DomGlobal.document); // Fails
Range range = documentRange.createRange(); // Never reaches here

当我改为使用 anJs.uncheckedCast()时,它会成功:

DocumentRange documentRange = Js.uncheckedCast(DomGlobal.document);
Range range = documentRange.createRange(); // Works

的文档Js.uncheckedCast()说:

“你应该总是更喜欢定期铸造而不是这个(除非你知道你在做什么!)。”

我不知道为什么我必须使用它,所以我感到很紧张。有人可以解释如何Js.cast()执行其类型检查以及为什么我需要Js.uncheckedCast()在这种情况下使用 an 吗?

4

1 回答 1

5

Js.cast()是一种欺骗的方式,并且做一些 Java 语言不允许但实际上可能是合法的事情。忽略“它实际上是如何工作的”,这个想法是你现在可以解决 Java 会抱怨的问题,即使它被证明是合法的。

一个示例可能是您使用 ajava.lang.Doubledouble希望将其视为 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()- 这里只有一个方法,而且不太可能以有意义的方式改变。

于 2020-05-26T16:54:37.633 回答