Как я могу изменить этот суперкласс на абстрактный класс без получения исключения экземпляра? - PullRequest
2 голосов
/ 25 октября 2019

Я хочу сделать мой суперкласс абстрактным классом, однако я получаю различные исключения при попытке сделать это (InstantiationException, RuntimeException, InvocationTargetException). Мой код работает, когда у меня нет суперкласса как абстрактного. Как мне это исправить?

Я попытался включить мои функции "main" и "start" вне тела абстрактного класса, так как я полагал, что это может быть проблема с вызовом этих функций внутри абстрактного класса. Я переместил эти методы в подкласс, который расширяет абстрактный класс. Затем я переопределил абстрактный метод «start» в каждом подклассе. Это переформатирование не решило ничего, хотя я получаю те же ошибки. Из того, что я исследовал в этом вопросе Как вы вызываете метод подкласса из суперкласса в Java? формат, который использует этот пользователь, близок к тому, что я пытаюсь достичь с помощью моего приложения "Расширение абстрактного суперкласса". "

package tutorial.application;

import javafx.application.Application;
import javafx.scene.Group;
import javafx.scene.Scene;
import javafx.scene.canvas.Canvas;
import javafx.scene.canvas.GraphicsContext;
import javafx.stage.Stage;
import javafx.scene.paint.Color;
import java.lang.Math;


public abstract class Test extends Application{

    private int width = 0;
    private int height = 0;

    public Test() {
        this.width = 0;
        this.height = 0;
    }

    public Test(int width, int height) {
        this.width = width;
        this.height = height;
    }

    public abstract void draw(GraphicsContext GC);


    public class MyLine extends Test{

        private int x1, x2, y1, y2;

        public MyLine(int x1, int y1, int x2, int y2) {
            super(0, 0);
            this.x1 = x1;
            this.x2 = x2;
            this.y1 = y1;
            this.y2 = y2;
        }

        @Override
        public void draw(GraphicsContext GC){
            GC.strokeLine(x1, y1, x2, y2);
        }
    }

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

    @Override
    public void start(Stage primaryStage) {

        int canvasx = 400, canvasy = 650;
        MyLine line = new MyLine(0, 0, canvasx, canvasy);
        MyLine line2 = new MyLine(0, canvasy, canvasx, 0);
        Group root = new Group();
        Canvas canvas = new Canvas(canvasx, canvasy);
        GraphicsContext GC = canvas.getGraphicsContext2D();
        line.draw(GC);
        line2.draw(GC);
        root.getChildren().add(canvas);
        primaryStage.setScene(new Scene(root));
        primaryStage.show();
    }

}

Когда у меня нет суперкласса в качестве абстрактного класса, а рисование больше не является абстрактным методом, я получаю следующий желаемый результат: https://i.imgur.com/J4XB8bo.png

Однаков противном случае я получаю следующие сообщения об ошибках:

Exception in Application constructor
java.lang.reflect.InvocationTargetException
    at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
    at java.base/jdk.internal.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    at java.base/java.lang.reflect.Method.invoke(Method.java:567)
    at javafx.graphics/com.sun.javafx.application.LauncherImpl.launchApplicationWithArgs(LauncherImpl.java:464)
    at javafx.graphics/com.sun.javafx.application.LauncherImpl.launchApplication(LauncherImpl.java:363)
    at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
    at java.base/jdk.internal.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    at java.base/java.lang.reflect.Method.invoke(Method.java:567)
    at java.base/sun.launcher.LauncherHelper$FXHelper.main(LauncherHelper.java:1051)
Caused by: java.lang.RuntimeException: Unable to construct Application instance: class tutorial.application.Test
    at javafx.graphics/com.sun.javafx.application.LauncherImpl.launchApplication1(LauncherImpl.java:890)
    at javafx.graphics/com.sun.javafx.application.LauncherImpl.lambda$launchApplication$2(LauncherImpl.java:195)
    at java.base/java.lang.Thread.run(Thread.java:835)
Caused by: java.lang.InstantiationException
    at java.base/jdk.internal.reflect.InstantiationExceptionConstructorAccessorImpl.newInstance(InstantiationExceptionConstructorAccessorImpl.java:48)
    at java.base/java.lang.reflect.Constructor.newInstanceWithCaller(Constructor.java:500)
    at java.base/java.lang.reflect.Constructor.newInstance(Constructor.java:481)
    at javafx.graphics/com.sun.javafx.application.LauncherImpl.lambda$launchApplication1$8(LauncherImpl.java:802)
    at javafx.graphics/com.sun.javafx.application.PlatformImpl.lambda$runAndWait$12(PlatformImpl.java:455)
    at javafx.graphics/com.sun.javafx.application.PlatformImpl.lambda$runLater$10(PlatformImpl.java:428)
    at java.base/java.security.AccessController.doPrivileged(AccessController.java:389)
    at javafx.graphics/com.sun.javafx.application.PlatformImpl.lambda$runLater$11(PlatformImpl.java:427)
    at javafx.graphics/com.sun.glass.ui.InvokeLaterDispatcher$Future.run(InvokeLaterDispatcher.java:96)
    at javafx.graphics/com.sun.glass.ui.win.WinApplication._runLoop(Native Method)
    at javafx.graphics/com.sun.glass.ui.win.WinApplication.lambda$runLoop$3(WinApplication.java:174)
    ... 1 more
Exception running application tutorial.application.Test

Редактировать: до сих пор я пробовал около 3 существенно разных вариантов, все из которых привели к ошибкам. Сначала я попытался сделать статический подкласс внутри иерархии следующим образом:

static class MyLine extends Test{
.
.
.
}

Я также попытался снова извлечь внутренний класс "MyLine" из суперкласса "Test" следующим образом:

public abstract class Test extends Application{

    private int width = 0;
    private int height = 0;

    public Test() {
        this.width = 0;
        this.height = 0;
    }

    public Test(int width, int height) {
        this.width = width;
        this.height = height;
    }

    public abstract void draw(GraphicsContext GC);

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

    @Override
    public void start(Stage primaryStage) {

        int canvasx = 400, canvasy = 650;
        MyLine line = new MyLine(0, 0, canvasx, canvasy);
        MyLine line2 = new MyLine(0, canvasy, canvasx, 0);
        Group root = new Group();
        Canvas canvas = new Canvas(canvasx, canvasy);
        GraphicsContext GC = canvas.getGraphicsContext2D();
        line.draw(GC);
        line2.draw(GC);
        root.getChildren().add(canvas);
        primaryStage.setScene(new Scene(root));
        primaryStage.show();
    }
}

class MyLine extends Test{

    private int x1, x2, y1, y2;

    public MyLine(int x1, int y1, int x2, int y2) {
        super(0, 0);
        this.x1 = x1;
        this.x2 = x2;
        this.y1 = y1;
        this.y2 = y2;
    }

    @Override
    public void draw(GraphicsContext GC){
        GC.strokeLine(x1, y1, x2, y2);
    }
}

Laslty Я разработал новый класс «Main» для расширения «Test» и запуска приложения оттуда следующим образом:

package tutorial.application;

import javafx.application.Application;
import javafx.scene.Group;
import javafx.scene.Scene;
import javafx.scene.canvas.Canvas;
import javafx.scene.canvas.GraphicsContext;
import javafx.stage.Stage;
import javafx.scene.paint.Color;
import java.lang.Math;


public abstract class Test extends Application{

    private int width = 0;
    private int height = 0;

    public Test() {
        this.width = 0;
        this.height = 0;
    }

    public Test(int width, int height) {
        this.width = width;
        this.height = height;
    }

    public abstract void draw(GraphicsContext GC);


}

class MyLine extends Test{

    private int x1, x2, y1, y2;

    public MyLine(int x1, int y1, int x2, int y2) {
        super(0, 0);
        this.x1 = x1;
        this.x2 = x2;
        this.y1 = y1;
        this.y2 = y2;
    }

    @Override
    public void draw(GraphicsContext GC){
        GC.strokeLine(x1, y1, x2, y2);
    }

    @Override
    public void start(Stage arg0) throws Exception {

    }
}

class Main extends Test{
    private Main() {
        super(0,0);
    }
    public static void main(String []args){
        launch(args);
    }

    @Override
    public void start(Stage primaryStage) {

        int canvasx = 400, canvasy = 650;
        MyLine line = new MyLine(0, 0, canvasx, canvasy);
        MyLine line2 = new MyLine(0, canvasy, canvasx, 0);
        Group root = new Group();
        Canvas canvas = new Canvas(canvasx, canvasy);
        GraphicsContext GC = canvas.getGraphicsContext2D();
        line.draw(GC);
        line2.draw(GC);
        root.getChildren().add(canvas);
        primaryStage.setScene(new Scene(root));
        primaryStage.show();
    }

    @Override
    public void draw(GraphicsContext GC) {

    }
}

Первые два по-прежнему приводят к тем же ошибкам, а последнийЯ считаю, что это ухудшает мою проблему, поскольку в потоке "main" выдается новое исключение:

Exception in thread "main" java.lang.reflect.InvocationTargetException

Ответы [ 2 ]

2 голосов
/ 25 октября 2019

Причина, по которой вы не можете создать экземпляр подкласса, если суперкласс является абстрактным, заключается в том, что этот подкласс является внутренним классом суперкласса, и для создания экземпляра внутреннего класса вам нужен экземпляр содержащего класса, нопо определению вы не можете создать экземпляр абстрактного класса.

Сделайте подкласс static, и ваша проблема должна быть решена.

0 голосов
/ 26 октября 2019

Я нашел рабочее решение, переименовав мой файл класса в «Main». Это позволило мне сохранить абстрактный класс «Test» и все его подклассы, и я запустил приложение через мой новый открытый класс «Main», который расширяет «Application».

...