1

在 javaWindows 10 pro x64 jre 1.8.0_60中,以下代码产生预期的输出(单击 html 按钮后):

Hello World

但是在javaWindows 10 pro x64 jre 1.8.0_152中似乎存在某种断开连接,因为单击按钮时它不会向控制台输出任何内容

为什么在最新版本的 java(当时是 152)上我的代码会给出不可预测且通常不需要的结果。我试图提供最少的代码来创建下面的场景。

import javafx.application.Application;
import javafx.beans.value.ChangeListener;
import javafx.beans.value.ObservableValue;
import javafx.concurrent.Worker;
import javafx.scene.Scene;
import javafx.scene.layout.BorderPane;
import javafx.scene.web.WebEngine;
import javafx.scene.web.WebView;
import javafx.stage.Stage;
import netscape.javascript.JSObject;

public class Main extends Application {

public static void main(String[] args) {
    launch(args);
}

@Override
public void start(Stage primaryStage) {
    WebView browser = new WebView();
    final WebEngine webEngine = browser.getEngine();
    webEngine.setJavaScriptEnabled(true);
    webEngine.load("https://api.ipify.org/?format=json");
    webEngine.getLoadWorker().stateProperty().addListener(
            new ChangeListener<Worker.State>() {
                @Override
                public void changed(ObservableValue ov, Worker.State oldState, Worker.State newState) {
                    if (newState == Worker.State.SUCCEEDED) {
                        JSObject jso = (JSObject) webEngine.executeScript("window");
                        webEngine.executeScript(
                                "var button = document.createElement(\"button\");\n" +
                                        "button.innerHTML = \"Do Something\";\n" +
                                        "var body = document.getElementsByTagName(\"body\")[0];\n" +
                                        "body.appendChild(button);\n" +
                                        "button.addEventListener (\"click\", function() {java.exit();});");
                        jso.setMember("java", new Bridge());

                    }
                }
            });
    BorderPane panel = new BorderPane(browser);
    Scene scene = new Scene(panel, 700, 700);
    primaryStage.setScene(scene);
    primaryStage.show();
}

public class Bridge {
    public void exit() {
        System.out.println("Hello World");
    }
}}
4

1 回答 1

1

您正在传递new Bridge()给 setMember 方法。由于没有变量保存 Bridge 实例,因此在您按下按钮之前它会被垃圾收集。

WebEngine 文档

请注意,在上面的示例中,应用程序持有对JavaApplication实例的引用。这是 JavaScript 回调执行所需方法所必需的。

在以下示例中,应用程序不包含对 Java 对象的引用:

JSObject window = (JSObject) webEngine.executeScript("window");
window.setMember("app", new JavaApplication());

在这种情况下,由于属性值是一个本地对象,"new JavaApplication()"所以该值可能会在下一次 GC 循环中被垃圾回收。

当用户点击链接时,并不保证会执行回调方法exit

将 Bridge 对象保存在字段中以防止对其进行垃圾收集:

new ChangeListener<Worker.State>() {
    private final Bridge bridge = new Bridge();

    @Override
    public void changed(ObservableValue ov, Worker.State oldState, Worker.State newState) {
        if (newState == Worker.State.SUCCEEDED) {

            // ...

            jso.setMember("java", bridge);
        }
    }

为什么在早期版本的 Java 中没有发生这种情况?因为不同的 Java 版本可以自由地改变垃圾收集的时间和行为。你很幸运,但在后来的版本中,你的运气用完了。

于 2017-11-11T03:06:14.300 回答