等待主框架加载时的 JxBrowser TimeoutException (invokeAndWaitFinishLoadingMainFrame)
JxBrowser TimeoutException when waiting for main frame to load (invokeAndWaitFinishLoadingMainFrame)
我在项目中使用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方法是问题所在:
- 第一个视图(登录视图)使用 invokeAndWaitFinishLoadingMainFrame 方法
正确加载
- 之后我的程序在再次使用 invokeAndWaitFinishLoadingMainFrame 方法时超时(下面是完全异常)
- 例如,当我点击登录按钮并希望转到主视图时
我无意中发现了一个小 hack 来让它暂时工作(现在它变得令人毛骨悚然):
- 当我加载在线网页时,超时问题消失了,例如google
- 也可以用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");
}
});
}
前面提到的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[=14=](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
总结一下:
无效:
- 加载登录视图
- 点击登录按钮
- 尝试加载主视图 -> 超时
有效:
- 加载登录视图
- 点击登录按钮
- 加载google
- 加载后直接google(来不及用户交互),加载主视图->无超时
这里发生了什么 :O ?
任何帮助都会很棒!
JxBrowser 基于 Chromium 引擎并继承其 multi-process 架构。每个 Browser
实例都与至少一个处理 DOM 和 Java 脚本相关功能的渲染进程相关联。当您导航到不同的域时,Chromium 引擎会创建一个新的渲染进程并终止旧的。如果您在一个域内导航,呈现过程保持不变。
从调用堆栈中,我看到您尝试从 DOM 侦听器中加载 URL。 DOM 侦听器是同步调用的,因此在 Java 处理事件时渲染进程保持阻塞状态。
当您加载 URL 时,例如google.com,一切正常,因为创建了一个新的渲染进程,一旦页面加载,Browser.invokeAndWaitFinishLoadingMainFrame
方法 returns 控件和旧的渲染进程被终止。但是,如果您尝试加载 file:///
URL,渲染过程保持不变并且无法开始加载,因为渲染过程被 Browser.invokeAndWaitFinishLoadingMainFrame
调用的方法阻塞 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();
}
我在项目中使用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方法是问题所在:
- 第一个视图(登录视图)使用 invokeAndWaitFinishLoadingMainFrame 方法 正确加载
- 之后我的程序在再次使用 invokeAndWaitFinishLoadingMainFrame 方法时超时(下面是完全异常)
- 例如,当我点击登录按钮并希望转到主视图时
我无意中发现了一个小 hack 来让它暂时工作(现在它变得令人毛骨悚然):
- 当我加载在线网页时,超时问题消失了,例如google
- 也可以用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"); } });
}
前面提到的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[=14=](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
总结一下:
无效:
- 加载登录视图
- 点击登录按钮
- 尝试加载主视图 -> 超时
有效:
- 加载登录视图
- 点击登录按钮
- 加载google
- 加载后直接google(来不及用户交互),加载主视图->无超时
这里发生了什么 :O ?
任何帮助都会很棒!
JxBrowser 基于 Chromium 引擎并继承其 multi-process 架构。每个 Browser
实例都与至少一个处理 DOM 和 Java 脚本相关功能的渲染进程相关联。当您导航到不同的域时,Chromium 引擎会创建一个新的渲染进程并终止旧的。如果您在一个域内导航,呈现过程保持不变。
从调用堆栈中,我看到您尝试从 DOM 侦听器中加载 URL。 DOM 侦听器是同步调用的,因此在 Java 处理事件时渲染进程保持阻塞状态。
当您加载 URL 时,例如google.com,一切正常,因为创建了一个新的渲染进程,一旦页面加载,Browser.invokeAndWaitFinishLoadingMainFrame
方法 returns 控件和旧的渲染进程被终止。但是,如果您尝试加载 file:///
URL,渲染过程保持不变并且无法开始加载,因为渲染过程被 Browser.invokeAndWaitFinishLoadingMainFrame
调用的方法阻塞 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();
}