gpt4 book ai didi

java - 等待主框架加载时出现 JxBrowser TimeoutException (invokeAndWaitFinishLoadingMainFrame)

转载 作者:行者123 更新时间:2023-12-02 09:57:04 37 4
gpt4 key购买 nike

我正在项目中使用 JxBrowser。我只使用本地 HTML 文件,因此使用以下方法来呈现我的 HTML 文件:

    public static void loadHTMLFile(Browser browser, String filename){
String current = PathUtils.getCurrentDir();
browser.loadURL("file:///" + current + "/some/path/in/my/project/resources/web/" + filename);
}

在某些情况下,我必须使用invokeAndWaitFinishLoadingMainFrame方法,因为我必须初始化必须首先完全加载的按钮。

来自 JxBrowser 官方网站 ( https://jxbrowser.support.teamdev.com/support/solutions/articles/9000013107-loading-waiting ) 的示例如下所示:

// Blocks current thread execution and waits until http://www.google.com web page is loaded completely
Browser.invokeAndWaitFinishLoadingMainFrame(browser, new Callback<Browser>() {
@Override
public void invoke(Browser value) {
value.loadURL("http://www.google.com");
}
});

就我而言,它看起来像这样:

public static void loadHTMLFileComplete(Browser browser, String filename){
Browser.invokeAndWaitFinishLoadingMainFrame(browser, new Callback<Browser>() {
@Override
public void invoke(Browser value) {
loadHTMLFile(value, filename);
}
});
}

到目前为止,我想没什么奇怪的......

现在是棘手的部分:

invokeAndWaitFinishLoadingMainFrame方法是问题所在:

  1. 使用 invokeAndWaitFinishLoadingMainFrame 方法正确加载第一个 View (登录 View )
  2. 之后,当再次使用 invokeAndWaitFinishLoadingMainFrame 方法时,我的程序超时(下面是完全异常)
    • 例如,当我单击登录按钮并希望转发到主视图时
  3. 我无意中发现了一个小技巧,让它暂时可以工作(现在它变得令人毛骨悚然):

    • 当我在中间加载在线网页时,超时问题消失了。谷歌
    • 也可以使用invokeAndWaitFinishLoadingMainFrame方法
    • 我就是这样做的

      public static void loadGoogle(Browser browser){
      Browser.invokeAndWaitFinishLoadingMainFrame(browser, new Callback<Browser>() {
      @Override
      public void invoke(Browser value) {
      value.loadURL("http://www.google.com");

      }
      });

      }

  4. 这里出现了我之前提到的 timeoutException:

    -- Product name: JxBrowser
    -- Licensed version: 6.x
    -- Licensed to:
    -- License type: Evaluation
    -- Generation date: 21.04.2019
    -- Expiration date: 21.05.2019
    -- License info: Single-user license
    -- Current date: 30.04.2019
    JxBrowser license valid.
    11:55:59 SCHWERWIEGEND:[0430/235559.671039:ERROR:sandbox_linux.cc(379)] InitializeSandbox() called with multiple threads in process gpu-process.
    com.teamdev.jxbrowser.chromium.Browser@6b9651f3
    11:56:49 SCHWERWIEGEND: Unexpected exception in DOMEventListener.handleEvent() method.
    java.lang.RuntimeException: java.util.concurrent.TimeoutException
    at com.teamdev.jxbrowser.chromium.Browser.invokeAndWaitFinishLoadingMainFrame(SourceFile:570)
    at com.teamdev.jxbrowser.chromium.Browser.invokeAndWaitFinishLoadingMainFrame(SourceFile:532)
    at browserActions.PageLoader.loadHTMLFileComplete(PageLoader.java:15)
    at browserViews.OrderCreationView.loadView(OrderCreationView.java:21)
    at browserActions.ButtonInitializer.lambda$0(ButtonInitializer.java:26)
    at com.teamdev.jxbrowser.chromium.dom.internal.EventTarget$a.onMessageReceived(SourceFile:179)
    at com.teamdev.jxbrowser.chromium.internal.ipc.q.a(SourceFile:1085)
    at com.teamdev.jxbrowser.chromium.internal.ipc.r.run(SourceFile:69)
    at com.teamdev.jxbrowser.chromium.internal.s.run(SourceFile:79)
    at java.util.concurrent.Executors$RunnableAdapter.call(Executors.java:511)
    at java.util.concurrent.FutureTask.run(FutureTask.java:266)
    at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1149)
    at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:624)
    at java.lang.Thread.run(Thread.java:748)
    Caused by: java.util.concurrent.TimeoutException
    ... 14 more

总结:

它不起作用:

  • 加载登录 View
  • 点击登录按钮
  • 尝试加载主视图 -> 超时

有效:

  • 加载登录 View
  • 点击登录按钮
  • 加载谷歌
  • 加载谷歌后立即加载(用户没有时间交互),加载主视图 -> 无超时

这里发生了什么:O?

任何帮助都会很棒!

最佳答案

JxBrowser基于Chromium引擎,继承了其多进程架构。每个 Browser 实例都与至少一个处理 DOM 和 JavaScript 相关功能的渲染进程相关联。当您导航到不同的域时,Chromium 引擎会创建一个新的渲染进程并终止旧的渲染进程。如果您在一个域内导航,渲染过程保持不变。

从调用堆栈中,我看到您尝试从 DOM 监听器中加载 URL。 DOM 监听器是同步调用的,因此在 Java 处理事件时渲染进程保持阻塞状态。

当您加载 URL 时,例如google.com,一切正常,因为创建了一个新的渲染进程,并且一旦加载页面,Browser.invokeAndWaitFinishLoadingMainFrame 方法就会返回控制权,并且旧的渲染进程将被终止。但是,如果您尝试加载 file:/// URL,渲染过程将保持不变,并且无法启动加载,因为渲染过程被 Browser.invokeAndWaitFinishLoadingMainFrame 阻止。 code> 从 DOM 监听器调用的方法,所以你会遇到死锁。

为了避免此异常,您不应从 DOM 监听器内启动同步导航。

根据调用堆栈,您从 DOMEventListener 调用 loadView 方法。为了避免死锁,您应该异步调用此方法,例如:

import com.teamdev.jxbrowser.chromium.dom.events.DOMEvent;
import com.teamdev.jxbrowser.chromium.dom.events.DOMEventListener;

import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;

public class MyDomEventListener implements DOMEventListener {

private final ExecutorService executorService;
private final OrderCreationView view;

public MyDomEventListener(OrderCreationView view) {
this.view = view;
this.executorService = Executors.newCachedThreadPool();
}

@Override
public void handleEvent(DOMEvent domEvent) {
// Do not block the current thread and invoke the loadView method asynchronously.
executorService.execute(view::loadView);
}
}

loadView 方法中,您可以同步加载 HTML:

public void loadView() {
Browser.invokeAndWaitFinishLoadingMainFrame(browser, new Callback<Browser>() {
@Override
public void invoke(Browser browser) {
browser.loadHTML("");
}
});
initLoginButton();
}

关于java - 等待主框架加载时出现 JxBrowser TimeoutException (invokeAndWaitFinishLoadingMainFrame),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/55929434/

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