我有一个方法,为WebView的WebEngine设置一个新的网页,需要等到页面加载完毕才能在当前方法中继续。
基本上我想要:
MyMethod()
{
navigateToNewUrl();
waitForPageToLoad(); // This is the part I am struggling with
doSomethingElse();
}
我尝试过使用ChangeListener()但它只会在我的方法执行完毕后执行。我用谷歌搜索了许多导致更多挫败感的术语,例如" java非阻塞等待布尔值"。最终我做到了启动新线程(以防止Application GUI锁定)和使用CountDownTimer(而不是Thread.join())。以下代码是我无法实现我想要的东西。
public static Boolean pageLoaded;
public volatile static CountDownLatch latch;
private static void TestMethod()
{
// Set the CountDownLatch
latch = new CountDownLatch(1);
// Set the page loaded flag
pageLoaded = false;
// Navigate to new window
myWebView.getEngine().load("https://www.google.com/");
// Call a method, that starts a thread, that checks if the page has loaded
WaitForPageLoaded();
// Wait for the CountDownLatch, latch
try { latch.await(); } catch (InterruptedException e) { /* TODO Auto-generated catch block */ e.printStackTrace(); }
// Write result to console
System.out.println("[pageLoaded] " + pageLoaded);
}
private static void WaitForPageLoaded()
{
Thread thread = new Thread()
{
public void run()
{
// 120 iterations * 250 ms = 30 seconds
for (int i = 0; i < 120; i++)
{
// Check if page has loaded
if (pageLoaded == true)
{
break;
}
else
{
System.out.println("Waited " + (i*250) + " milliseconds, so far...");
try { Thread.sleep(250); }
catch (InterruptedException e) { /* TODO Auto-generated catch block */ e.printStackTrace(); }
}
}
// Tell the calling method to move on with life
latch.countDown();
}
}
}
在另一个.java文件(类)中,我有:
// Get the WebEngine and set a listener for a state change
WebEngine webEngine = webView.getEngine();
webEngine.getLoadWorker().stateProperty().addListener
(
new ChangeListener<State>()
{
public void changed(@SuppressWarnings("rawtypes") ObservableValue ov, State oldState, State newState)
{
if (newState == State.SUCCEEDED)
{
// Set the page loaded flag
OtherClassName.pageLoaded = true;
}
}
}
);
答案 0 :(得分:3)
与几乎所有UI框架一样,JavaFX基于事件驱动模型。在幕后有一个循环运行(在FX应用程序线程中)处理事件。你不应该直接处理这个循环。
您正尝试通过重新创建此循环并等待条件来解决问题,而不是注册响应相应事件的处理程序。对于要加载的页面,您只需要响应每个页面完成加载并执行测试过程中的下一步,而不是在一个线程中等待(哪个线程没有明显的方法来执行此操作)。您可以通过在Web引擎stateProperty
上注册侦听器来完成此操作。
所以你的结构看起来应该是
public class MyApp extends Application {
private WebView webView ;
private WebEngine engine ;
// variables representing the current state of your testing process...
@Override
public void start(Stage primaryStage) {
webView = new WebView();
engine = webView.getEngine();
engine.getLoadWorker().stateProperty().addListener((obs, oldState, newState) -> {
if (newState == Worker.State.SUCCEEDED) {
// new page has loaded, process:
testMethod();
}
});
// set up UI etc...
engine.load(...); // load first page
}
private void testMethod() {
// check current state, do next appropriate thing, update variables...
}
}
此结构完全符合您的要求 - 重复调用testMethod()
,但仅在每个页面完成加载时才会调用 - 但框架会为您管理“等待”。