如何在Eclipse上解决java.lang.非法访问异常(JavaFX)

5w9g7ksd  于 2023-03-18  发布在  Eclipse
关注(0)|答案(2)|浏览(198)

我尝试通过根据功能创建单独的包来整理应用程序的文件,但遇到此错误:

java.lang.IllegalAccessException: class javafx.fxml.FXMLLoader$ValueElement (in module javafx.fxml) cannot access class application.login.Login (in module eSentral_Desktop_App) because module eSentral_Desktop_App does not export application.login to module javafx.fxml

我的问题是如何解决这个问题,是什么原因导致它发生?
以下是一些额外信息:
这是我的类层次结构:

这是我的Main.java代码:

package application;
import javafx.application.Application;
import javafx.fxml.FXMLLoader;
import javafx.stage.Stage;
import jfxtras.styles.jmetro.JMetro;
import jfxtras.styles.jmetro.Style;
import javafx.scene.Parent;
import javafx.scene.Scene;
import java.io.IOException;
import application.login.Users;
import application.listing.Homepage;
public class Main extends Application{
    public static Object currentUser = new Object();
    double x,y = 0;
    private static Stage stg;
    @Override
    public void start(Stage primaryStage) throws Exception{
        try {
            stg = primaryStage;
            Parent root = FXMLLoader.load(getClass().getResource("login/LoginUI.fxml"));
            Scene scene = new Scene(root);
            JMetro jMetro = new JMetro(scene, Style.LIGHT);
            scene.getStylesheets().add(getClass().getResource("application.css").toExternalForm());
            
            primaryStage.setScene(scene);
            primaryStage.show();
            
        } catch(Exception e) {
            e.printStackTrace();
        }
    }
    public void changeScene(String fxml) throws IOException {
        Parent pane = FXMLLoader.load(getClass().getResource(fxml));
        stg.getScene().setRoot(pane);
    }
    public static void main(String[] args) {
        launch(args);
    }
}

这是我的LoginUI.fxml代码:

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

<?import javafx.scene.control.Button?>
<?import javafx.scene.control.Label?>
<?import javafx.scene.control.PasswordField?>
<?import javafx.scene.control.Separator?>
<?import javafx.scene.control.TextField?>
<?import javafx.scene.image.Image?>
<?import javafx.scene.image.ImageView?>
<?import javafx.scene.layout.AnchorPane?>
<?import javafx.scene.layout.BorderPane?>
<?import javafx.scene.text.Font?>

<BorderPane maxHeight="-Infinity" maxWidth="-Infinity" prefHeight="700.0" prefWidth="800.1" xmlns="http://javafx.com/javafx/17" xmlns:fx="http://javafx.com/fxml/1" fx:controller="application.login.Login">
   <top>
      <AnchorPane maxHeight="-Infinity" maxWidth="-Infinity" prefHeight="115.0" prefWidth="800.0" BorderPane.alignment="CENTER">
         <children>
            <ImageView fitHeight="80.0" fitWidth="249.0" layoutX="276.0" layoutY="18.0" pickOnBounds="true" preserveRatio="true">
               <image>
                  <Image url="@../resources/ic_eSentral_Logo.png" />
               </image>
            </ImageView>
         </children>
      </AnchorPane>
   </top>
   <bottom>
      <AnchorPane maxHeight="-Infinity" maxWidth="-Infinity" prefHeight="133.0" prefWidth="800.0" BorderPane.alignment="CENTER">
         <children>
            <Button fx:id="btnRegister" layoutX="248.0" layoutY="63.0" mnemonicParsing="false" onMouseEntered="#MouseEnter" onMouseExited="#MouseExit" prefHeight="44.0" prefWidth="318.0" style="-fx-background-color: #169fe3;" text="Register a new account" textFill="WHITE">
               <font>
                  <Font name="System Bold" size="20.0" />
               </font>
            </Button>
            <Label alignment="CENTER" layoutX="311.0" layoutY="24.0" text="Don't have an account yet?" textFill="#1286bc">
               <font>
                  <Font name="System Bold" size="15.0" />
               </font>
            </Label>
         </children>
      </AnchorPane>
   </bottom>
   <center>
      <AnchorPane maxHeight="-Infinity" maxWidth="-Infinity" prefHeight="487.0" prefWidth="800.0" BorderPane.alignment="CENTER">
         <children>
            <Label fx:id="lblSuccess" alignment="CENTER" layoutX="247.0" layoutY="2.0" prefHeight="24.0" prefWidth="318.0" textFill="#00ff26">
               <font>
                  <Font name="System Bold" size="15.0" />
               </font>
            </Label>
            <TextField fx:id="txtUsername" layoutX="247.0" layoutY="77.0" prefHeight="44.0" prefWidth="318.0" promptText="Email">
               <font>
                  <Font size="15.0" />
               </font>
            </TextField>
            <Label fx:id="lblEmail" alignment="CENTER" layoutX="247.0" layoutY="121.0" prefHeight="24.0" prefWidth="318.0" textFill="#ec1e1e">
               <font>
                  <Font name="System Bold" size="15.0" />
               </font>
            </Label>
            <PasswordField fx:id="txtPassword" layoutX="247.0" layoutY="145.0" prefHeight="44.0" prefWidth="318.0" promptText="Password">
               <font>
                  <Font size="15.0" />
               </font>
            </PasswordField>
            <Label fx:id="lblPassword" alignment="CENTER" layoutX="247.0" layoutY="189.0" prefHeight="24.0" prefWidth="318.0" textFill="#ec1e1e">
               <font>
                  <Font name="System Bold" size="15.0" />
               </font>
            </Label>
            <Label fx:id="lblForgot" alignment="CENTER" layoutX="326.0" layoutY="226.0" onMouseClicked="#forgetPassword" onMouseEntered="#MouseEnter" onMouseExited="#MouseExit" text="Forget your password?  " textFill="#1286bc" underline="true">
               <font>
                  <Font size="15.0" />
               </font>
            </Label>
            <Label fx:id="lblForgot1" alignment="CENTER" layoutX="380.0" layoutY="35.0" text="Login" textFill="#1286bc">
               <font>
                  <Font name="System Bold" size="20.0" />
               </font>
            </Label>
            <Button fx:id="btnLogin" layoutX="247.0" layoutY="274.0" mnemonicParsing="false" onAction="#login" onMouseEntered="#MouseEnter" onMouseExited="#MouseExit" prefHeight="44.0" prefWidth="318.0" style="-fx-background-color: #169fe3;" text="Log in" textFill="WHITE">
               <font>
                  <Font name="System Bold" size="20.0" />
               </font>
            </Button>
            <Button fx:id="btnFB" alignment="CENTER" layoutX="247.0" layoutY="380.0" mnemonicParsing="false" onMouseEntered="#MouseEnter" onMouseExited="#MouseExit" prefHeight="44.0" prefWidth="318.0" style="-fx-background-color: #4766a9;" text="Log in with Facebook" textFill="WHITE">
               <font>
                  <Font name="System Bold" size="20.0" />
               </font>
               <graphic>
                  <ImageView fitHeight="35.0" fitWidth="63.0" pickOnBounds="true" preserveRatio="true">
                     <image>
                        <Image url="@../resources/ic_facebook.png" />
                     </image>
                  </ImageView>
               </graphic>
            </Button>
            <Separator layoutX="247.0" layoutY="338.0" prefHeight="28.0" prefWidth="130.0" />
            <Separator layoutX="434.0" layoutY="338.0" prefHeight="28.0" prefWidth="130.0" />
            <Label layoutX="392.0" layoutY="337.0" text="OR" textFill="#1286bc">
               <font>
                  <Font name="System Bold" size="20.0" />
               </font>
            </Label>
         </children>
      </AnchorPane>
   </center>
</BorderPane>

这是模块信息代码:

module eSentral_Desktop_App {
    requires javafx.controls;
    requires java.net.http;
    requires javafx.fxml;
    requires javafx.base;
    requires javafx.graphics;
    requires com.google.gson;
    requires java.logging;
    requires java.desktop;
    requires org.jfxtras.styles.jmetro;
    
    opens application to javafx.graphics, javafx.fxml;
}
bfhwhh0e

bfhwhh0e1#

谢谢jewelsea,这个问题已经解决了。我需要做的就是添加

opens application.login to javafx.fxml;

到模块-info.java中。
它看起来像这样:

module eSentral_Desktop_App {
    requires javafx.controls;
    requires java.net.http;
    requires javafx.fxml;
    requires javafx.base;
    requires javafx.graphics;
    requires com.google.gson;
    requires java.logging;
    requires java.desktop;
    requires org.jfxtras.styles.jmetro;
    
    opens application.login to javafx.fxml;
    opens application to javafx.graphics, javafx.fxml;
}
myzjeezk

myzjeezk2#

TL;DR:

至少有两种解决方案:
1.在你的module-info.java文件中添加一个opens application.login to javafx.fxml指令。对于那些不是OP的人,用你自己的控制器包替换application.login
1.通过删除module-info.java文件使您的代码非模块化。在这种情况下,我建议将JavaFX保留在module-path中。如果您不知道如何做到这一点,请参阅Getting Started with JavaFX 11+。如果您不将JavaFX保留在module-path中,您将得到this warning,因为从技术上讲,JavaFX不支持将其放置在类路径中。

非法访问异常

此错误与Java 9中添加的 *Java平台模块系统 *(JPMS)有关。如果您不熟悉模块,我建议您阅读Understanding Java 9 Modules以了解基础知识。
JPMS的目标之一是为Java添加更强大的封装,这意味着:
1.模块A中的包的public1成员只能由模块B中的代码访问,前提是该包由模块A导出到 * 至少 * 个模块B。导出包是通过模块描述符(module-info.java文件)中的exports指令完成的。
注意 exported 包的公共成员也可以通过reflection访问。
1.模块A中的包的私有2成员只能由模块B中的代码反射访问,前提是该包由模块A打开到 * 至少 * 个模块B。打开包通过模块描述符中的opens指令完成。

错误消息

错误消息的格式为:

class C (in module A) cannot access class D (in module B) because module B does not export P to module A

其中类D是你的类,在你的模块中,BP是类D的包,注意类C在这里并不重要。
如果错误消息显示“does notexport",则修复方法是添加:

module B {
    // other directives

    // add this
    exports P to A; // where 'to A' is optional (qualified vs unqualified exports)
}

**警告:**在尝试使用合格的exports之前,请务必阅读下面的“解决方案”部分。

而如果错误消息显示“does notopen",则修复方法是添加:

module B {
    // other directives

    // add this
    opens P to A; // where 'to A' is optional (qualified vs unqualified opens)
}

注意有可能这个错误首先提到没有 * 导出 * 一个包,然后在你添加了一个exports指令后抱怨没有 * 打开 * 一个包,这取决于使用反射的代码是如何实现的。

**1.**这意味着public类和所述public类的任何public成员(即,方法、字段等)。注意,在非公共类中声明的public.e.,g方法不被认为是“公共成员”(至少在此上下文中)。
2.“public members”未包含的任何内容,即protected、package-private和private。注意,如果要反射调用非public成员,则必须调用setAccessible(true)(或trySetAccessible()并检查返回值)。

溶液

在您的例子中,错误是关于application.login包中的一个类,下面的例子就是基于这个类。
你的错误说,“does notexport"。这意味着,从我上面描述的,你应该添加一个exports指令到你的模块描述符。这将是好的,当且仅当:

  • 没有注入FXML的字段,或者所有这些字段都是public
  • 您没有initialize()方法,或者该方法是public
  • 没有从FXML文件链接的事件处理程序方法,或者这些方法都是public

如果您满足这些要求,则具有:

// WARNING: Does not actually work with JavaFX
exports application.login to javafx.fxml; // a qualified exports

**至少,如果JavaFX除了正常的JPMS检查之外,不执行自己的检查(针对方法),那么这是真的。**不管出于什么原因,JavaFX决定在通过反射调用方法时,不接受限定导出。您必须使用 unqualified 导出:

exports aplication.login;

但是,很少有FXML控制器能够满足上面列出的所有要求。如果没有FXML注入字段、initialize()方法和事件处理程序方法,那么为什么要首先使用控制器?如果确实有其中任何一个,它们真的应该都是private,因为它们都是控制器的实现细节。因此,正确的解决方案是添加opens指令:

// JavaFX is perfectly fine with qualified opens, just not qualified exports
opens application.login to javafx.fxml;

就好像错误消息说“不打开"。
FXML简介对此进行了详细说明:

将应用程序部署为模块

如果使用FXMLLoader加载命名模块中的类型,则应用程序必须确保FXML文件中引用的所有类型(包括控制器类和任何自定义Node类)均可由javafx.fxml模块反射访问。如果模块opens包含至少javafx.fxml模块的包,则类型可反射访问。
例如,如果com.foo.MyController位于foo.app模块中,则module-info.java可能如下所示:

module foo.app {
    opens com.foo to javafx.fxml;
}

或者,如果模块exports无条件地包含包,则类型是反射可访问的。

  • @javafx.fxml.FXML注解也记录了这一点。

替代解决方案

另一种解决方案是让你的代码非模块化。你可以从你的项目中删除module-info.java文件。这可能是更好的,特别是当你的代码 * 直接 * 依赖于非模块化依赖项时。
也就是说,如果你让你的代码非模块化,我建议你把JavaFX放在模块路径上,如果你不这样做,你会看到下面Q&A中描述的警告:

如果您不知道如何确保JavaFX位于模块路径上,请参见Getting Started with JavaFX 11+

相关问题