groovy - 使用 WebLogic InitialContext 后无法从 FXML 打开新窗口



这是一个非常非常奇怪和特殊的情况。我做了一个最小的可验证的例子:

文件:Main.groovy

public class Main extends Application {

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

@Override
public void start(Stage primaryStage) {
Button destroy = new Button("Break the program")
Button openWindow= new Button("Open new Window")
destroy.setOnAction(new EventHandler<ActionEvent>() {
@Override public void handle(ActionEvent e) {
Hashtable<String, String> env = new Hashtable<String, String>();
env.put(Context.INITIAL_CONTEXT_FACTORY, "weblogic.jndi.WLInitialContextFactory");
env.put(Context.PROVIDER_URL, "[IP censored]");
InitialContext ic = new InitialContext(env);
}
});
openWindow.setOnAction(new EventHandler<ActionEvent>() {
@Override public void handle(ActionEvent e) {
FXMLLoader loader = new FXMLLoader();
loader.setLocation(getClass().getClassLoader().getResource("test.fxml"))
TestController testController =new TestController()
loader.setController(testController)
Parent testWindowRoot = loader.load();
Scene scene = new Scene(testWindowRoot);

Stage stage = new Stage()
stage.setTitle("Test");
stage.setScene(scene);
stage.show();
}
});
HBox hbox = new HBox(destroy,openWindow)
StackPane root = new StackPane();
root.setPadding(new Insets(5));
root.getChildren().add(hbox);
primaryStage.setTitle("JavaFX Test");
primaryStage.setScene(new Scene(root, 300, 150));
primaryStage.show();
}
}

文件:TestController.groovy

class TestController implements Initializable {

@FXML AnchorPane mainAnchorPane
@FXML Label label

public void initialize(URL arg0, ResourceBundle arg1) {
label.setText("test 2") //this is line 20
}

}

文件:test.fxml

<?xml version="1.0" encoding="UTF-8"?>
<?import javafx.scene.control.Label?>
<?import javafx.scene.layout.AnchorPane?>
<AnchorPane fx:id="mainAnchorPane" maxHeight="-Infinity" maxWidth="-Infinity" minHeight="-Infinity" minWidth="-Infinity" prefHeight="400.0" prefWidth="600.0" xmlns="http://javafx.com/javafx/8.0.171" xmlns:fx="http://javafx.com/fxml/1">
<children>
<Label fx:id="label" layoutX="240.0" layoutY="192.0" text="Test 1" />
</children>
</AnchorPane>

所以这是一个非常基本的例子来重现发生的事情。当我单击"打开窗口"按钮时,我会看到一个新窗口,文本为Test 2,因为在TestController的initialize()方法中有一个对setText("Test 2")的调用。然后,当我点击"中断程序"时,它会创建一个映射,并用JMS信息填充它(很抱歉,我不能给你JMS队列的IP(,并实例化InitialContext。之后,任何单击"打开窗口"按钮的操作都会使程序崩溃。没有其他窗口打开过。我看到以下堆栈跟踪:

java.lang.NullPointerException: Cannot invoke method setText() on null object
[...]
at TestController.initialize(TestController.groovy:20)

这当然不是真的,因为在我尝试连接到WebLogic之前,他打开那个窗口从来没有遇到过问题。

注意:只有当我从Groovy执行JavaFX时才会发生这种情况,而不是从Java执行JavaFX,这对我来说已经完美地工作了几个月。

那怎么回事?这两件事之间有什么联系?为什么在实例化InitialContext后无法打开FXML窗口?

使用jdk1.8.0_181

编辑:我找到了一个变通办法!不过,它只是解决了症状。问题是GUI对象在使用WL后被神奇地设置为null。好吧,我可以使用lookup((方法来"刷新"对象引用。这显然不是一个完美的解决方案,但我的代码现在可以工作了。不过,我仍然非常好奇这些问题的真实答案。

我认为这是因为你创建了这样的控制器

TestController testController =new TestController()

使用loader

FXMLLoader loader = new FXMLLoader(getClass().getClassLoader().getResource("test.fxml"));
TestController testController = (TestController) loader.getController();

TestController testController =  loader.getController();

只是一个猜测:初始化WebLogic可能会扰乱类加载器。在使用之前,请尝试打印出getClass().getClassLoader().getResource("test.fxml")的值。

相关内容

最新更新