java - 映射后加载 fxml 时,spring 出现 javafx 应用程序异常

标签 java spring spring-boot javafx fxml

如果没有映射 fxml,此代码可以完美运行并加载 fxml

如果我映射 fxml 出现异常 我尝试了很多,但无法理解问题所在......

SpringFxmlLoader 类方法 load() return 语句不返回任何内容,因此我认为发生异常

javafx.fxml.LoadException:/E:/aa/JavaFXwithSpringBoot/bin/test/spring/boot/LoginPage.fxml:15

at javafx.fxml.FXMLLoader.constructLoadException(FXMLLoader.java:2601) at javafx.fxml.FXMLLoader.loadImpl(FXMLLoader.java:2579) at javafx.fxml.FXMLLoader.loadImpl(FXMLLoader.java:2441) at javafx.fxml.FXMLLoader.load(FXMLLoader.java:2409) at test.spring.boot.config.SpringFXMLLoader.load(SpringFXMLLoader.java:34) at test.spring.boot.config.StageManager.loadViewNodeHierarchy(StageManager.java:80) at test.spring.boot.config.StageManager.switchScene(StageManager.java:31) at test.spring.boot.Main.displayInitialScene(Main.java:35) at test.spring.boot.Main.start(Main.java:24) at com.sun.javafx.application.LauncherImpl.lambda$launchApplication1$162(LauncherImpl.java:863) at com.sun.javafx.application.PlatformImpl.lambda$runAndWait$175(PlatformImpl.java:326) at com.sun.javafx.application.PlatformImpl.lambda$null$173(PlatformImpl.java:295) at java.security.AccessController.doPrivileged(Native Method) at com.sun.javafx.application.PlatformImpl.lambda$runLater$174(PlatformImpl.java:294) at com.sun.glass.ui.InvokeLaterDispatcher$Future.run(InvokeLaterDispatcher.java:95) at com.sun.glass.ui.win.WinApplication._runLoop(Native Method) at com.sun.glass.ui.win.WinApplication.lambda$null$148(WinApplication.java:191) at java.lang.Thread.run(Unknown Source) Caused by: org.springframework.beans.factory.NoSuchBeanDefinitionException: No qualifying bean of type 'test.spring.boot.LoginPageController' available at org.springframework.beans.factory.support.DefaultListableBeanFactory.getBean(DefaultListableBeanFactory.java:353) at org.springframework.beans.factory.support.DefaultListableBeanFactory.getBean(DefaultListableBeanFactory.java:340) at org.springframework.context.support.AbstractApplicationContext.getBean(AbstractApplicationContext.java:1090) at javafx.fxml.FXMLLoader$ValueElement.processAttribute(FXMLLoader.java:929) at javafx.fxml.FXMLLoader$InstanceDeclarationElement.processAttribute(FXMLLoader.java:971) at javafx.fxml.FXMLLoader$Element.processStartElement(FXMLLoader.java:220) at javafx.fxml.FXMLLoader$ValueElement.processStartElement(FXMLLoader.java:744) at javafx.fxml.FXMLLoader.processStartElement(FXMLLoader.java:2707) at javafx.fxml.FXMLLoader.loadImpl(FXMLLoader.java:2527) ... 16 more Exception in Application start method 2017-10-24 17:18:03.779 ERROR 6416 --- [lication Thread] test.spring.boot.config.StageManager : Unable to load FXML view >> /test/spring/boot/LoginPage.fxml

org.springframework.beans.factory.NoSuchBeanDefinitionException: No qualifying bean of type 'test.spring.boot.LoginPageController' available at org.springframework.beans.factory.support.DefaultListableBeanFactory.getBean(DefaultListableBeanFactory.java:353) ~[spring-beans-4.3.11.RELEASE.jar:4.3.11.RELEASE] at org.springframework.beans.factory.support.DefaultListableBeanFactory.getBean(DefaultListableBeanFactory.java:340) ~[spring-beans-4.3.11.RELEASE.jar:4.3.11.RELEASE] at org.springframework.context.support.AbstractApplicationContext.getBean(AbstractApplicationContext.java:1090) ~[spring-context-4.3.11.RELEASE.jar:4.3.11.RELEASE] at javafx.fxml.FXMLLoader$ValueElement.processAttribute(FXMLLoader.java:929) ~[jfxrt.jar:na] at javafx.fxml.FXMLLoader$InstanceDeclarationElement.processAttribute(FXMLLoader.java:971) ~[jfxrt.jar:na] at javafx.fxml.FXMLLoader$Element.processStartElement(FXMLLoader.java:220) ~[jfxrt.jar:na] at javafx.fxml.FXMLLoader$ValueElement.processStartElement(FXMLLoader.java:744) ~[jfxrt.jar:na] at javafx.fxml.FXMLLoader.processStartElement(FXMLLoader.java:2707) ~[jfxrt.jar:na] at javafx.fxml.FXMLLoader.loadImpl(FXMLLoader.java:2527) ~[jfxrt.jar:na] at javafx.fxml.FXMLLoader.loadImpl(FXMLLoader.java:2441) ~[jfxrt.jar:na] at javafx.fxml.FXMLLoader.load(FXMLLoader.java:2409) ~[jfxrt.jar:na] at test.spring.boot.config.SpringFXMLLoader.load(SpringFXMLLoader.java:34) ~[bin/:na] at test.spring.boot.config.StageManager.loadViewNodeHierarchy(StageManager.java:80) [bin/:na] at test.spring.boot.config.StageManager.switchScene(StageManager.java:31) [bin/:na] at test.spring.boot.Main.displayInitialScene(Main.java:35) [bin/:na] at test.spring.boot.Main.start(Main.java:24) [bin/:na] at com.sun.javafx.application.LauncherImpl.lambda$launchApplication1$162(LauncherImpl.java:863) [jfxrt.jar:na] at com.sun.javafx.application.PlatformImpl.lambda$runAndWait$175(PlatformImpl.java:326) ~[jfxrt.jar:na] at com.sun.javafx.application.PlatformImpl.lambda$null$173(PlatformImpl.java:295) ~[jfxrt.jar:na] at java.security.AccessController.doPrivileged(Native Method) ~[na:1.8.0_131] at com.sun.javafx.application.PlatformImpl.lambda$runLater$174(PlatformImpl.java:294) ~[jfxrt.jar:na] at com.sun.glass.ui.InvokeLaterDispatcher$Future.run(InvokeLaterDispatcher.java:95) ~[jfxrt.jar:na] at com.sun.glass.ui.win.WinApplication._runLoop(Native Method) ~[jfxrt.jar:na] at com.sun.glass.ui.win.WinApplication.lambda$null$148(WinApplication.java:191) ~[jfxrt.jar:na] at java.lang.Thread.run(Unknown Source) ~[na:1.8.0_131]

2017-10-24 17:18:03.846 INFO 6416 --- [lication Thread] s.c.a.AnnotationConfigApplicationContext : Closing org.springframework.context.annotation.AnnotationConfigApplicationContext@f096164: startup date [Tue Oct 24 17:18:02 IST 2017]; root of context hierarchy 2017-10-24 17:18:03.848 INFO 6416 --- [lication Thread] o.s.j.e.a.AnnotationMBeanExporter : Unregistering JMX-exposed beans on shutdown java.lang.reflect.InvocationTargetException at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method) at sun.reflect.NativeMethodAccessorImpl.invoke(Unknown Source) at sun.reflect.DelegatingMethodAccessorImpl.invoke(Unknown Source) at java.lang.reflect.Method.invoke(Unknown Source) at com.sun.javafx.application.LauncherImpl.launchApplicationWithArgs(LauncherImpl.java:389) at com.sun.javafx.application.LauncherImpl.launchApplication(LauncherImpl.java:328) at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method) at sun.reflect.NativeMethodAccessorImpl.invoke(Unknown Source) at sun.reflect.DelegatingMethodAccessorImpl.invoke(Unknown Source) at java.lang.reflect.Method.invoke(Unknown Source) at sun.launcher.LauncherHelper$FXHelper.main(Unknown Source) Caused by: java.lang.RuntimeException: Exception in Application start method at com.sun.javafx.application.LauncherImpl.launchApplication1(LauncherImpl.java:917) at com.sun.javafx.application.LauncherImpl.lambda$launchApplication$155(LauncherImpl.java:182) at java.lang.Thread.run(Unknown Source) Caused by: java.lang.NullPointerException: Root cannot be null at javafx.scene.Scene.(Scene.java:336) at javafx.scene.Scene.(Scene.java:194) at test.spring.boot.config.StageManager.prepareScene(StageManager.java:62) at test.spring.boot.config.StageManager.show(StageManager.java:36) at test.spring.boot.config.StageManager.switchScene(StageManager.java:32) at test.spring.boot.Main.displayInitialScene(Main.java:35) at test.spring.boot.Main.start(Main.java:24) at com.sun.javafx.application.LauncherImpl.lambda$launchApplication1$162(LauncherImpl.java:863) at com.sun.javafx.application.PlatformImpl.lambda$runAndWait$175(PlatformImpl.java:326) at com.sun.javafx.application.PlatformImpl.lambda$null$173(PlatformImpl.java:295) at java.security.AccessController.doPrivileged(Native Method) at com.sun.javafx.application.PlatformImpl.lambda$runLater$174(PlatformImpl.java:294) at com.sun.glass.ui.InvokeLaterDispatcher$Future.run(InvokeLaterDispatcher.java:95) at com.sun.glass.ui.win.WinApplication._runLoop(Native Method) at com.sun.glass.ui.win.WinApplication.lambda$null$148(WinApplication.java:191) ... 1 more Exception running application test.spring.boot.Main

Main.java

@ComponentScan("test.spring.boot")
@SpringBootApplication
public class Main extends Application {

    protected ConfigurableApplicationContext springContext;
    protected StageManager stageManager;

    @Override
    public void start(Stage primaryStage) throws IOException {
        stageManager = springContext.getBean(StageManager.class, primaryStage);
        displayInitialScene();
    }

    @Override
    public void init() throws Exception
    {
        springContext=springBootApplicationContext();
    }


    protected void displayInitialScene() {
        stageManager.switchScene(FxmlView.LOGIN);
    }


     @Override
     public void stop() throws Exception {
        springContext.close();
     }

    private ConfigurableApplicationContext springBootApplicationContext() {
        SpringApplicationBuilder builder = new SpringApplicationBuilder(Main.class);
        String[] args = getParameters().getRaw().stream().toArray(String[]::new);
        return builder.run(args);
    }

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

Controller .java

public class LoginPageController implements Initializable{

    @FXML
    private TextField txtUserName;

    @FXML
    private PasswordField txtPassward;

    @FXML
    private Button btnLogin;

    @FXML
    private Label lblLogin; 

    @Autowired
    private UserService userService;

    @Lazy
    @Autowired
    private StageManager stageManager;

    @FXML
    public void login()
    {
        if(userService.authenticate(txtUserName.getText(), txtPassward.getText()))
        {
            stageManager.switchScene(FxmlView.USER);
        }
        else
        {
            lblLogin.setText("Login Failed.");
        }
    }


    @Override
    public void initialize(URL location, ResourceBundle resources) {

    }
}

FxmlView.java

public enum FxmlView {
    USER {
        @Override
        public String getTitle() {
            return getStringFromResourceBundle("user.title");
        }

        @Override
        public String getFxmlFile() {
            return "/test/spring/boot/TestSpring.fxml";
        }
    }, 


    LOGIN {
        @Override
        public String getTitle() {
            return getStringFromResourceBundle("login.title");
        }

        @Override
        public String getFxmlFile() {
            return "/test/spring/boot/LoginPage.fxml";
        }
    };

    public abstract String getTitle();
    public abstract String getFxmlFile();

    String getStringFromResourceBundle(String key){
        return ResourceBundle.getBundle("Bundle").getString(key);
    }
}

SpringFxmlLoader.java

@Component
public class SpringFXMLLoader {
    private final ResourceBundle resourceBundle;
    private final ApplicationContext context;

    @Autowired
    public SpringFXMLLoader(ApplicationContext context, ResourceBundle resourceBundle) {
        this.resourceBundle = resourceBundle;
        this.context = context;
    }

    public Parent load(String fxmlPath) throws IOException {      
        FXMLLoader loader = new FXMLLoader();
        loader.setControllerFactory(context::getBean); //Spring now FXML Controller Factory
        loader.setResources(resourceBundle);
        loader.setLocation(getClass().getResource(fxmlPath));
        System.out.println("springfxmlloader class : "+fxmlPath);
        System.out.println("LOADER : ");
        return loader.load();
    }
}

StageManager.java

public class StageManager {

 private static final Logger LOG = getLogger(StageManager.class);
    private final Stage primaryStage;
    private final SpringFXMLLoader springFXMLLoader;

    public StageManager(SpringFXMLLoader springFXMLLoader, Stage stage) {

    System.out.println(">>>>>>>>>>>>>>>>>> 1");

        this.springFXMLLoader = springFXMLLoader;
        this.primaryStage = stage;
    }

    public void switchScene(final FxmlView view) {
        System.out.println(">>>>>>>>>>>>>>>>>> 2");
        Parent viewRootNodeHierarchy = loadViewNodeHierarchy(view.getFxmlFile());
        show(viewRootNodeHierarchy, view.getTitle());
    }

    private void show(final Parent rootnode, String title) {
        Scene scene = prepareScene(rootnode);
        //scene.getStylesheets().add("/styles/Styles.css");

        //primaryStage.initStyle(StageStyle.TRANSPARENT);
        primaryStage.setTitle(title);
        primaryStage.setScene(scene);
        primaryStage.sizeToScene();
        primaryStage.centerOnScreen();

        System.out.println(">>>>>>>>>>>>>>>>>> 3");

        try {
            primaryStage.show();
        } catch (Exception exception) {
            logAndExit ("Unable to show scene for title" + title,  exception);
        }
    }

    private Scene prepareScene(Parent rootnode){

        System.out.println(">>>>>>>>>>>>>>>>>> 4");
        Scene scene = primaryStage.getScene();

        System.out.println("SCN >>>>>>>>>>>>>>>>>>>> : "+ scene);

        if (scene == null) {
            scene = new Scene(rootnode);
        }
        scene.setRoot(rootnode);
        return scene;
    }


    private Parent loadViewNodeHierarchy(String fxmlFilePath) {

        System.out.println(">>>>>>>>>>>>>>>>>> 5 path: "+fxmlFilePath);
        Parent rootNode = null;
        try {
            System.out.println(fxmlFilePath);
            rootNode = springFXMLLoader.load(fxmlFilePath);
            System.out.println(">>>>5 : try");
            Objects.requireNonNull(rootNode, "A Root FXML node must not be null");
        } catch (Exception exception) {

            System.out.println("File path????? : "+fxmlFilePath);

            logAndExit("Unable to load FXML view >> " + fxmlFilePath, exception);
            exception.printStackTrace();
        }

        return rootNode;
    }


    private void logAndExit(String errorMsg, Exception exception) {

        System.out.println(">>>>>>>>>>>>>>>>>> 6");
        LOG.error(errorMsg, exception, exception.getCause());
        Platform.exit();
    }
}

最佳答案

您忘记添加 Stereotype在您的 LoginPageController 类上,这就是 Spring 告诉您的原因:

org.springframework.beans.factory.NoSuchBeanDefinitionException: No qualifying bean of type 'test.spring.boot.LoginPageController'

由于该类未标记,组件扫描将忽略它,并且不会为该类型分配任何 bean 寄存器。

尝试:

@Controller
public class LoginPageController implements Initializable{

    @Autowired
    public LoginPageController(UserService userService, private StageManager stageManager){
        this.userService = userService;
        this.stageManager = stageManager;
    }
}

关于java - 映射后加载 fxml 时,spring 出现 javafx 应用程序异常,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/46910304/

相关文章:

java - JDB 中 GDB 'start' 的等价物?

java - 使用Hystrix通过断路器从Eureka获取服务列表

spring - 失败时重试逻辑 - Spring Reactor

java - @Autowire 调用期间未实例化服务类

java - NullPointerException 甚至所有对象都已初始化

java - Java阻塞线程占用CPU资源多吗?

java - 使用 Gradle 从 CLI 运行 Spring Boot 测试

java - 不同类中的两个相同变量

oracle - 如何将Oracle配置为Spring引导应用程序

spring - Spring Boot 将 '' 下的属性绑定(bind)到 com.zaxxer.hikari.HikariDataSource 失败