gwt - DevMode 启动在 GWT JSNI 项目中失败

标签 gwt jsni gwt-super-dev-mode dev-mode

我在 gwt 开发模式调试中遇到了一个奇怪的问题。

以下是我正在编写的 JSNI 包装器 https://github.com/sillysachin/GWTAMChart

这是一个相当小且简单的项目,包含大量 JSNI、JavaScriptObject 和 JSON 代码。它覆盖了流行的 amcharts 图表库。它在 SuperDevMode 和生产环境中调试时运行良好。

但是我无法使用开发模式调试在 Internet Explorer 中调试项目。

java.lang.ClassFormatError: 类文件 com/google/gwt/core/client/JavaScriptObject$ 中的重复方法名称和签名

抛出的主要异常没有帮助我确定代码的哪一部分被破坏了!!!!!

java.lang.ClassFormatError: Duplicate method name&signature in class file com/google/gwt/core/client/JavaScriptObject$
    at java.lang.ClassLoader.defineClass1(Native Method)
    at java.lang.ClassLoader.defineClass(ClassLoader.java:800)
    at java.lang.ClassLoader.defineClass(ClassLoader.java:643)
    at com.google.gwt.dev.shell.CompilingClassLoader.findClass(CompilingClassLoader.java:1142)
    at com.google.gwt.dev.shell.CompilingClassLoader.loadClass(CompilingClassLoader.java:1215)
    at java.lang.ClassLoader.loadClass(ClassLoader.java:358)
    at java.lang.Class.forName0(Native Method)
    at java.lang.Class.forName(Class.java:270)
    at com.google.gwt.dev.shell.JsValueGlue.set(JsValueGlue.java:220)
    at com.google.gwt.dev.shell.ModuleSpaceOOPHM.doInvoke(ModuleSpaceOOPHM.java:130)
    at com.google.gwt.dev.shell.ModuleSpace.invokeNative(ModuleSpace.java:589)
    at com.google.gwt.dev.shell.ModuleSpace.invokeNativeVoid(ModuleSpace.java:315)
    at com.google.gwt.dev.shell.ModuleSpace.onLoad(ModuleSpace.java:359)
    at com.google.gwt.dev.shell.OophmSessionHandler.loadModule(OophmSessionHandler.java:200)
    at com.google.gwt.dev.shell.BrowserChannelServer.processConnection(BrowserChannelServer.java:530)
    at com.google.gwt.dev.shell.BrowserChannelServer.run(BrowserChannelServer.java:368)
    at java.lang.Thread.run(Thread.java:745)

最佳答案

有问题的类是 AmChartJSO 实现了 IsAmChart 接口(interface) - 所有方法都在 JavaScriptObject$ 中声明了两次。字节码片段:

public final synthetic com_amcharts_api_IsAmChart_setVersion(Ljava/lang/String;)V
  ALOAD 0
  ALOAD 1
  INVOKESTATIC com/amcharts/jso/AmChartJSO$.setVersion$ (Lcom/amcharts/jso/AmChartJSO;Ljava/lang/String;)V
  RETURN
  MAXSTACK = 2
  MAXLOCALS = 2

// access flags 0x1011
public final synthetic com_amcharts_api_IsAmChart_setVersion(Ljava/lang/String;)V
  ALOAD 0
  ALOAD 1
  INVOKESTATIC com/amcharts/jso/AmChartJSO$.setVersion$ (Lcom/amcharts/jso/AmChartJSO;Ljava/lang/String;)V
  RETURN
  MAXSTACK = 2
  MAXLOCALS = 2

您似乎遇到了覆盖类型的限制 - only one JavaScriptObject subtype can implement any given interface :

Practically speaking, this means that only one JavaScriptObject type may implement any given interface, but any number of non-JavaScriptObject types may also implement that interface.

看看你的代码,这个限制被打破了:AmChartJSO 实现了 IsAmChart,但是 AmCoordinateChartJSO 实现了 IsAmCoordinateChart extends IsAmChart - 因此两个 JSO 实现相同的接口(interface)。如果我对这个限制的理解正确,你甚至不能将实现接口(interface)的 JSO 子类化。

我做了一个快速测试,这段代码也失败了:

public class Test extends JavaScriptObject implements TakesValue<String> {
    protected Test() {
    }

    @Override
    public final void setValue(String value) {
    }

    @Override
    public final String getValue() {
        return null;
    }
}

public class Test2 extends Test {
    protected Test2() {
    }
}

还有一个同样无用的异常(exception):

java.lang.NullPointerException: null
    at com.google.gwt.dev.shell.CompilingClassLoader$MySingleJsoImplData.findOverloadUsingErasure(CompilingClassLoader.java:703)
    at com.google.gwt.dev.shell.CompilingClassLoader$MySingleJsoImplData.<init>(CompilingClassLoader.java:593)
    at com.google.gwt.dev.shell.CompilingClassLoader.<init>(CompilingClassLoader.java:980)
    at com.google.gwt.dev.shell.ShellModuleSpaceHost.onModuleReady(ShellModuleSpaceHost.java:137)
    at com.google.gwt.dev.shell.ModuleSpace.onLoad(ModuleSpace.java:340)
    at com.google.gwt.dev.shell.OophmSessionHandler.loadModule(OophmSessionHandler.java:200)
    at com.google.gwt.dev.shell.BrowserChannelServer.processConnection(BrowserChannelServer.java:526)
    at com.google.gwt.dev.shell.BrowserChannelServer.run(BrowserChannelServer.java:364)
    at java.lang.Thread.run(Thread.java:745)

请参阅this thread on GWT's mailing list有关此问题的解决方法和一般性讨论。


自己转储生成的类文件

为了将来引用,您可以通过设置 gwt.dev.classDump 系统属性 (-Dgwt.dev.classDump=true) 转储生成的 class 文件)。参见 this wiki page想要查询更多的信息。默认情况下,类被写入 rewritten-classes 文件夹(在您的情况下它将是 war/rewritten-classes)。这些类按包组织,因此查找 JavaScriptObject$ 很容易:rewritten-classes/com/google/gwt/core/client/JavaScriptObject$.class

现在,您需要做的就是拆解它 - 我已经使用了 Bytecode Outline plugin for Eclipse得到JavaScriptObject$.class的字节码。

要找出哪些方法是重复的,我可以用类加载器加载 class 文件,然后让 JVM 找出来...但是我觉得很懒,所以我只是 grep在字节码中搜索public final synthetic并运行uniq -D仅查看重复的条目。

关于gwt - DevMode 启动在 GWT JSNI 项目中失败,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/27269138/

相关文章:

java - 扩展不同基类的类包装器

java - GWT 将 InlineLabels 添加到带有文本的段落或 div

intellij-idea - Gwt super 开发模式调试器中的 java.util.Lists

java - GWT 服务器端反射

GWT CellTable 列排序

gwt - JSNI - 从另一个 native 方法调用 native 方法

java - 如何使用 Javascript 覆盖类型从 JSON 数组构建对象数组?

gwt - 将 Java 数组传递到 JavaScript(通过 JSNI)并返回到 Java 结果为空值

GWT 2.7 super 开发模式在同一网络上测试时不工作

maven - GWT Super Dev Mode 不编译最新资源