Java 无法使用@FXML ANOTION访问超类控制器中的元素

Java 无法使用@FXML ANOTION访问超类控制器中的元素,java,javafx,fxml,Java,Javafx,Fxml,我正在尝试使用JavaFX为我的程序建立一个简单的示例。 我想要的是一个带有main.fxml的控制器;然后main.fxml将有一个TabbedPane作为根目录,有两个选项卡(tab1.fxml和tab2.fxml),每个选项卡都有其控制器(Tab1Controller和Tab2Controller) 这可能是个问题,但我不明白为什么会有问题: 选项卡1控制器和选项卡2控制器都扩展控制器;因为它们共享各种被操纵的字段,例如需要不断更新的底部状态栏 到目前为止,我设法使所有控制器和.fxml都

我正在尝试使用JavaFX为我的程序建立一个简单的示例。 我想要的是一个带有main.fxml的控制器;然后main.fxml将有一个TabbedPane作为根目录,有两个选项卡(tab1.fxml和tab2.fxml),每个选项卡都有其控制器(Tab1Controller和Tab2Controller)

这可能是个问题,但我不明白为什么会有问题:

选项卡1控制器和选项卡2控制器都扩展控制器;因为它们共享各种被操纵的字段,例如需要不断更新的底部状态栏

到目前为止,我设法使所有控制器和.fxml都正常工作

当我试图从其中一个子类在控制器中设置标签的文本时,它只指向null,但标签在gui中以其默认文本初始化。。 编译正确,并且lblController似乎正确链接到main.fxml中的fx:id

非常感谢任何帮助/链接

我一直在看各种帖子,但唯一一篇接近我需要的帖子是这篇:

主要内容:

主控制器:

package sample;

import javafx.fxml.FXML;
import javafx.scene.control.Label;

public class Controller {
    @FXML
    public Label lblController;

    public Controller() {
        System.out.println("CONTROLLER MAIN - CONSTRUCTOR");
    }

    @FXML
    public void initialize() {
        System.out.println("CONTROLLER MAIN - INITIALIZER");
    }

    protected void setSts(String sts) {
        lblController.setText(sts);
    }
}
main.fxml

<?xml version="1.0" encoding="UTF-8"?>

<?import javafx.scene.control.Label?>
<?import javafx.scene.control.Tab?>
<?import javafx.scene.control.TabPane?>
<?import javafx.scene.layout.*?>
<AnchorPane xmlns:fx="http://javafx.com/fxml/1"
            xmlns="http://javafx.com/javafx/8.0.60"
            fx:controller="sample.Controller">
    <BorderPane>
        <center>
            <TabPane tabClosingPolicy="UNAVAILABLE">
                <Tab text="Tab 1">
                    <AnchorPane>
                        <!--FOR fx:id HAS TO HAVE THE SAME NAME BUT IN LOWER CASE AS THE .fxml-->
                        <fx:include fx:id="tab1" source="tab/tab1.fxml"/>
                    </AnchorPane>
                </Tab>
                <Tab text="Tab 2">
                    <AnchorPane>
                        <fx:include fx:id="tab2" source="tab/tab2.fxml"/>
                    </AnchorPane>
                </Tab>
            </TabPane>
        </center>
        <bottom>
            <AnchorPane>
                <Label fx:id="lblController" text="Controller Test"/>
            </AnchorPane>
        </bottom>
    </BorderPane>
</AnchorPane>
tab1.fxml

<?xml version="1.0" encoding="UTF-8"?>

<?import javafx.scene.control.Button?>
<?import javafx.scene.control.Label?>
<?import javafx.scene.control.TextField?>
<?import javafx.scene.layout.AnchorPane?>
<AnchorPane xmlns:fx="http://javafx.com/fxml/1"
            xmlns="http://javafx.com/javafx/8.0.60" fx:controller="sample.ctrTab.Tab1Controller">
    <children>
        <Label fx:id="lbl1" layoutX="100" layoutY="50" text="Def tab 1"/>
        <TextField fx:id="txt1" layoutX="100" layoutY="70"/>
        <Button fx:id="btn1Save" onAction="#btn1Save" layoutX="100" layoutY="140" mnemonicParsing="false"
                text="Save text"/>
        <Button fx:id="btn1Load" onAction="#btn1Load" layoutX="200.0" layoutY="140" mnemonicParsing="false"
                text="Send to tab 2"/>
    </children>
</AnchorPane>

这些字段可能存在于控制器类中,但是
fxmloader
仍然会为给定这些fxml加载的每个fxml创建新的控制器实例。由于
tab1.fxml
不包含带有
fx:id=“lblController”
属性的
Label
标记,因此不会将
lblController
字段注入
Tab1Controller
实例

最好不要扩展
控制器
,而是将其引用传递给子控制器:

public class Controller {
    @FXML
    public Label lblController;
    @FXML
    private Tab1Controller tab1Controller;
    @FXML
    private Tab2Controller tab2Controller;

    public Controller() {
        System.out.println("CONTROLLER MAIN - CONSTRUCTOR");
    }

    @FXML
    public void initialize() {
        System.out.println("CONTROLLER MAIN - INITIALIZER");
        tab1Controller.initParentController(this);
        tab2Controller.initParentController(this);
    }

    public void setSts(String sts) {
        lblController.setText(sts);
    }
}

请注意,对于子控制器,您实际上可以使用抽象超类实现
initParentController
方法一次。

我见过这种方法,我试图分别为每个子控制器避免initialize();它可以设置主控制器。我仍然很困惑继承在这里是如何不起作用的,即使使用分离的getter/setter进行访问。@4673j加载
main.fxml
,您只需获得3个不同的控制器实例:每个使用的fxml文件一个。我经常看到没有经验的fxml文件用户希望字段“神奇地”被注入到任何地方;事实并非如此。对象仅注入到与fxml文件一起使用的控制器实例中,在该实例中声明对象。我理解您的意思,我不希望使用magic,我希望了解它如何工作以及如何正确使用它,我不理解如果元素已按照注入请求初始化,我如何无法访问它?因此应用简单的继承;对于一个简单的括号,在我的程序(不是这个示例)中,我已经有9个控制器,并且还在增长@4673_j您似乎混淆了“对象/实例”和“类”。代码创建了三个不同的控制器实例。一个是
控制器
的实例,一个是
选项卡1控制器
的实例,一个是
选项卡2控制器
的实例。使用您定义的继承,后两个当然也是
Controller
的实例,因此这三个都有一个名为
lblController
的字段。然而,只有当FXML文件中有一个带有
fx:id=“lblController”
的元素时,该字段才被初始化,这仅在第一个实例中是正确的,而在其他两个实例中不是正确的。@James\u D这让事情更清楚了,我似乎忘记了基本内容;回到书上。谢谢你的努力。
<?xml version="1.0" encoding="UTF-8"?>

<?import javafx.scene.control.Button?>
<?import javafx.scene.control.Label?>
<?import javafx.scene.control.TextField?>
<?import javafx.scene.layout.AnchorPane?>
<AnchorPane xmlns:fx="http://javafx.com/fxml/1"
            xmlns="http://javafx.com/javafx/8.0.60" fx:controller="sample.ctrTab.Tab1Controller">
    <children>
        <Label fx:id="lbl1" layoutX="100" layoutY="50" text="Def tab 1"/>
        <TextField fx:id="txt1" layoutX="100" layoutY="70"/>
        <Button fx:id="btn1Save" onAction="#btn1Save" layoutX="100" layoutY="140" mnemonicParsing="false"
                text="Save text"/>
        <Button fx:id="btn1Load" onAction="#btn1Load" layoutX="200.0" layoutY="140" mnemonicParsing="false"
                text="Send to tab 2"/>
    </children>
</AnchorPane>
public class Controller {
    @FXML
    public Label lblController;
    @FXML
    private Tab1Controller tab1Controller;
    @FXML
    private Tab2Controller tab2Controller;

    public Controller() {
        System.out.println("CONTROLLER MAIN - CONSTRUCTOR");
    }

    @FXML
    public void initialize() {
        System.out.println("CONTROLLER MAIN - INITIALIZER");
        tab1Controller.initParentController(this);
        tab2Controller.initParentController(this);
    }

    public void setSts(String sts) {
        lblController.setText(sts);
    }
}
public class Tab1Controller {
    @FXML
    Label lbl1;
    @FXML
    TextField txt1;
    @FXML
    Button btn1Save, btn1Load;

    public Tab1Controller() {
        System.out.println("CONTROLLER TAB 1 - CONSTRUCTOR");
    }

    private Controller parentController;

    public void initParentController(Controller controller) {
         parentController = controller;
    }

    @FXML
    void btn1Save() {
        System.out.println("CONTROLLER TAB 1 - SAVE CLICK");
        parentController.lblController.setText("ANYTHING");
        parentController.setSts(txt1.getText());
    }

    @FXML
    void btn1Load() {
        System.out.println("CONTROLLER TAB 1 - LOAD CLICK");
    }

}