JavaFX在继续之前等待页面加载

时间:2015-05-11 21:51:17

标签: multithreading javafx-8

我有一个方法,为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;
                    }
                }
            }
        );

1 个答案:

答案 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(),但仅在每个页面完成加载时才会调用 - 但框架会为您管理“等待”。