gpt4 book ai didi

java - 即使在 vm args 中使用 XstartOnFirstThread,线程访问也无效

转载 作者:搜寻专家 更新时间:2023-11-01 02:29:32 26 4
gpt4 key购买 nike

我有一个只有一个类的初级 Java Web Start 应用程序。它可以在 Windows 和 Linux 上运行,但在 Mac OS X 上会出现可怕的无效线程访问错误。我意识到这已在其他地方得到处理。上网查了整整两天,所有的解决办法都执行了,但是问题依旧。

我的理解是,对 SWT 的调用必须从主线程进行,这里就是这种情况。如果我错了,请纠正我。

我将在下面发布 3 个片段,应用程序的源代码、jnlp 文件的相关部分以及 Mac 上的错误消息。问题在最后。


Java 源代码

package client;
import org.eclipse.swt.widgets.Display;
import org.eclipse.swt.widgets.Shell;
public class AccountWindow {
public static void main(String[] args) {
Display display = new Display(); **// error occurs here**
Shell shell = new Shell(display); shell.open();
while (!shell.isDisposed()) {
if (!display.readAndDispatch())
display.sleep();
}
display.dispose();
}
}

JNLP 片段

<resources os="Mac\ OS\ X" arch="x86_64">
<j2se version="1.5+" java-vm-args="-XstartOnFirstThread" />
<nativelib href="swt-4.2-cocoa-macosx-x86_64.jar" />
</resources>

错误信息

org.eclipse.swt.SWTException: Invalid thread access
at org.eclipse.swt.SWT.error(Unknown Source)
at org.eclipse.swt.SWT.error(Unknown Source)
at org.eclipse.swt.SWT.error(Unknown Source)
at org.eclipse.swt.widgets.Display.error(Unknown Source)
at org.eclipse.swt.widgets.Display.createDisplay(Unknown Source)
at org.eclipse.swt.widgets.Display.create(Unknown Source)
at org.eclipse.swt.graphics.Device.<init>(Unknown Source)
at org.eclipse.swt.widgets.Display.<init>(Unknown Source)
at org.eclipse.swt.widgets.Display.<init>(Unknown Source)
at client.AccountWindow.main(AccountWindow.java:16)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:39)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:25)
at java.lang.reflect.Method.invoke(Method.java:597)
at com.sun.javaws.Launcher.executeApplication(Launcher.java:1550)
at com.sun.javaws.Launcher.executeMainClass(Launcher.java:1488)
at com.sun.javaws.Launcher.doLaunchApp(Launcher.java:1299)
at com.sun.javaws.Launcher.run(Launcher.java:114)
at java.lang.Thread.run(Thread.java:637)

请注意
- 在 http://www.eclipse.org/swt/faq.php#javawebstart 发布的 display.syncExec 解决方案不适用,因为在您可以调用它之前,您需要一个显示。当我尝试创建显示时会发生这里的错误。
- 我已经使用 JaNeLa 验证了 jnlp 文件并且没有红色错误。
- 被正确解释,因为正在加载正确的 swt 库。
- 您可以在 http://thelinkjuicer.com/gannonline/client.jnlp 重现错误


现在是问题
任何人都可以在源代码或 jnlp 片段中看到任何会导致错误的内容吗?
次要问题:如何判断虚拟机是否实际读取了 -XstartOnFirstThread 参数?

最佳答案

很明显,您的 main 方法没有在主线程上执行。您可以在堆栈跟踪中看到启动器实际上是在另一个线程中启动的,然后是 Launcher。仅间接调用 main .不幸的是,这只是诊断,我不确定解决方案。我做过类似的事情(通过 Java Web Start 的 SWT 应用程序),但我不记得我们是如何解决这个问题的(如果有的话)。

检查 com.sun.javaws.Launcher 后源代码,目前尚不清楚如何使其工作。 Launcher.launch方法启动一个新线程,其中你的 main方法被执行。您可以按照代码重新创建您获得的确切堆栈跟踪。

main entry point Java Web Start 显示主线程在启动后很快就死了。

更新

我挖出了一些东西:在this Eclipse bug report建议问题可能与此有关:

<resources>
<j2se version="1.4+" />
<jar href="client.jar" />
</resources>

解析器从这里获取 j2se 规范并忽略后面更具体的规范。尝试删除 <j2se...行。

更新2

现在我从 here 中找到了这个:

com.apple.concurrent.Dispatch.getInstance().getNonBlockingMainQueueExecutor().execute(
new Runnable() { public void run() {
final Display display = Display.getDefault();
while (!display.isDisposed()) {
if (!display.readAndDispatch())
display.sleep();
}
});

这听起来确实可行。它完全按照我在下面的评论中描述的那样进行:通过专门为此目的而设置的机制将补丁添加到主线程中。尝试根据您的需要进行调整。您甚至可能不需要 -XstartOnFirstThread。

更新3

我终于找到了我的旧 SWT-JWS 项目。里面有这个:

<resources os="Mac OS X" arch="x86_64">
<j2se version="1.6+" java-vm-args="-XstartOnFirstThread"/>
<jar href="swt-cocoa-macosx-x86-64-3.6.2.jar" />
</resources>

并且有效。它没有默认 j2se元素,此元素出现在特定于 OSX 的条目中。

关于java - 即使在 vm args 中使用 XstartOnFirstThread,线程访问也无效,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/12937762/

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