gpt4 book ai didi

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

转载 作者:行者123 更新时间:2023-12-04 12:16:04 26 4
gpt4 key购买 nike

我在 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/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com