javac утверждает, что я не переопределяю метод в абстрактной реализации класса, когда я явно

я сделаю это как можно короче и по существу, но это своего рода сложная проблема. Я пишу на Java на платформе Linux, чего бы это ни стоило.

краткая версия цели: я хочу иметь абстрактный класс под названием Client это действует как общий контейнер для клиентских подключений. Client должен нанизывать каждое из своих соединений. У меня также есть полупроверенный код, который воспроизводит серверный аналог этого аналогичным образом. Абстрактное Client должны реализовываться в чем-то более ощутимым и instanceable. В моем случае у меня есть класс FileClientGui которая расширяет Client и переопределяет все Clientабстрактные методы с основным методом получения содержимого файла с сервера и их отображения. Это еще более осложняется тем, что абстрактное Client само по себе расширение java.lang.Thread.

Итак, вот моя файловая структура в общих терминах:

/ class / path/lib/клиент / клиент.java

/ class / path / com/fileclient / FileClientGui.java

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

поэтому я запускаю эту длинную команду javac на терминале, настраивая каталог classpath и build и все соответствующие файлы, которые должны быть составлено также. Единственная ошибка, которую я получаю для любого из этого кода, это:

com/fileclient/FileClientGui.java:26: com.fileclient.FileClientGui is not abstract and does not override abstract method cleanClients() in lib.client.Client

мой код (см. ниже) четко реализует метод и все другие абстрактные методы, определенные в клиент.Ява. Я прочесал Интернет, и кажется, что большинство людей, которые сталкиваются с этой ошибкой, пытаются сделать что-то вроде реализации ActionListener и запутаться с этой реализацией, и много раз, это просто проблема орфографии или капитализации. Я снова и снова повторял свой код, чтобы убедиться. что это не просто такая "ой" проблема. Я подозреваю, что это на самом деле какое-то столкновение между именем моего класса и именем какого-то другого класса, который каким-то образом оказался в моем пути к классам или в родной структуре/библиотеках Java, но я не могу найти ничего очевидного.

во всяком случае, вот мой код.

клиент.java:

package lib.client;

import lib.clientservercore.Connection;
import lib.simplefileaccess.Logger;
import java.io.IOException;
import java.net.Socket;
import java.util.ArrayList;
import java.lang.Thread;

/**
 *
 * @author Ryan Jung
 */
public abstract class Client extends Thread {

    ArrayList<Connection> connections;
    boolean isRunning;
    Logger log;

    public Client (String logFile) {
        log = new Logger(logFile);

        log.write("Initializing client...");

        connections = new ArrayList<Connection>(50);

        log.write("Client initialized.");
    }

    public void logOut(String contents) {
        log.write(contents);
    }

    public Logger getLogger() {
        return this.log;
    }

    public ArrayList<Connection> getConnections() {
        return connections;
    }

    public void addConnection(Connection c) {
        connections.add(c);
    }

    public void removeConnection(Connection c) {
        connections.remove(c);
    }

    public boolean getIsRunning() {
        return isRunning;
    }

    public void setIsRunning(boolean r) {
        isRunning = r;
    }

    public Connection connect(String host, int port) {
        log.write("Creating new connection...");

        Socket s;
        Connection c = null;

        // Validate port
        if (port <= 1024 || port > 65536) {
            log.write("Invalid server port: " + port + ".  Using 12321.");
            port = 12321;
        }

        try {
            s = new Socket(host, port);
            c = connectClient(s);
        } catch (IOException exIo) {
            log.write("Could not connect to the server at " + host + ":" + port + ".  Exception: " + exIo.getMessage());
            exIo.printStackTrace();
        }

        log.write("Connected client to " + host + ":" + port);

        return c;
    }

    @Override
    public void run() {
        log.write("Running client.");
        runClient();
        log.write("Client finished running.");
    }

    abstract Connection connectClient(Socket sock);

    abstract void runClient();

    abstract void cleanClients();

}

FileClientGui.java:

package com.fileclient;

import lib.client.Client;
import lib.clientservercore.Connection;
import lib.clientservercore.Connection.ConnectionStatus;
import java.awt.BorderLayout;
import java.awt.FlowLayout;
import java.awt.event.ActionListener;
import java.awt.event.ActionEvent;
import java.io.IOException;
import java.net.Socket;
import java.net.UnknownHostException;
import java.util.Iterator;
import javax.swing.JButton;
import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.JPanel;
import javax.swing.JTabbedPane;
import javax.swing.JTextField;
import java.lang.Thread;

/**
 *
 * @author Ryan Jung
 */
public class FileClientGui extends Client {

    JFrame frmMain;
    JPanel pnlMain;
    JPanel pnlConnect;
    JTabbedPane tabConnections;
    JLabel lblHost;
    JLabel lblPort;
    JTextField txtHost;
    JTextField txtPort;
    JButton btnConnect;

    public FileClientGui(String logFile) {
        super(logFile);

        logOut("Initializing client controller...");

        frmMain = new JFrame("Client");
        pnlMain = new JPanel(new BorderLayout());
        pnlConnect = new JPanel(new FlowLayout());
        tabConnections = new JTabbedPane();
        lblHost = new JLabel("Host:");
        lblPort = new JLabel("Port:");
        txtHost = new JTextField("localhost", 10);
        txtPort = new JTextField("12321", 5);
        btnConnect = new JButton("Connect");

        frmMain.setSize(450, 600);
        frmMain.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        frmMain.add(pnlMain);
        pnlMain.add(pnlConnect, BorderLayout.NORTH);
        pnlMain.add(tabConnections, BorderLayout.CENTER);
        pnlConnect.add(lblHost);
        pnlConnect.add(txtHost);
        pnlConnect.add(lblPort);
        pnlConnect.add(txtPort);
        pnlConnect.add(btnConnect);

        btnConnect.addActionListener(
            new ActionListener() {
                public void actionPerformed(ActionEvent e) {
                    String host = txtHost.getText();
                    int port = Integer.parseInt(txtPort.getText());
                    try {
                        Socket sock = new Socket(host, port);
                        FileClientConnectionGui c = (FileClientConnectionGui)(connectClient(sock));
                        tabConnections.addTab(c.getInetAddress().toString(), c.getMainPanel());
                    } catch (UnknownHostException ex) {
                        logOut("Can't find host: " + host + ".  Exception: " + ex.getMessage());
                        ex.printStackTrace();
                    } catch (IOException ex) {
                        logOut("Exception: " + ex.getMessage());
                        ex.printStackTrace();
                    }
                }
            }
        );

        frmMain.setVisible(true);

        logOut("Client controller initialized.");

    }

    public void removeConnection(FileClientConnectionGui c) {
        logOut("Removing connection: " + c.getInetAddress().toString());
        tabConnections.remove(c.getMainPanel());
        logOut("Removed connection.");
    }

    Connection connectClient(Socket sock) {
        logOut("Client controller is creating a new connection...");
        FileClientConnectionGui c = new FileClientConnectionGui(sock, getLogger(), this);
        addConnection(c);
        c.start();
        logOut("Client controller created a new connection.");
        return c;
    }

    void runClient() {
        setIsRunning(true);
        logOut("Client controller is running.");

        while (getIsRunning()) {
            cleanClients();
            try {
                sleep(500);
            } catch (InterruptedException ex) {
                logOut("Sleep interrupted.  Exception: " + ex.getMessage());
                ex.printStackTrace();
            }
        }

        logOut("Client controller stopped running.");
    }

    void cleanClients() {
        Iterator i = getConnections().iterator();
        try {
            while (i.hasNext()) {
                FileClientConnectionGui c = (FileClientConnectionGui)(i.next());
                if (c.getStatus() == ConnectionStatus.CLOSED) {
                    logOut("Removing dead client at " + c.getInetAddress().toString());
                    tabConnections.remove(c.getMainPanel());
                    removeConnection(c);
                }
            }
        } catch (Exception ex) {
            logOut("cleanClients Exception: " + ex.getMessage());
        }
    }

}

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

возможно, что больше всего сбивает с толку в этом (и, возможно, это дает ключ к проблеме?) заключается в том, что я могу комментировать другие реализации абстрактных методов (например, runClient или connectClient), и я не получаю никаких дополнительных проблем, просто тот же самый. Кроме того, если я добавлю директиву @Override к одному из этих других, как это:

@Override
Connection connectClient(Socket sock) {
    logOut("Client controller is creating a new connection...");
    FileClientConnectionGui c = new FileClientConnectionGui(sock, getLogger(), this);
    addConnection(c);
    c.start();
    logOut("Client controller created a new connection.");
    return c;
}

я вам еще ошибка:

com/fileclient/FileClientGui.java:96: method does not override or implement a method from a supertype

это понятно is переопределение метода от его супертипа (который является клиентом). Я попытался заменить "клиент" на полный classpath (lib.клиент.Клиента) и ни одна из ошибок не изменился.

я что-то упускаю? Что-то, чего я не пытаюсь?

1 ответов


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

вот простая пара классов, которые воспроизводят проблему:

package x1;

public abstract class P1
{
    abstract void foo();
}

и затем:

package x2;

public class P2 extends x1.P1
{
    void foo() {}
}

сборка их дает:

P2.java:3: P2 is not abstract and does not override abstract method foo() in P1
public class P2 extends x1.P1
       ^
1 error

делая foo protected в обоих классах устраняет проблему.