Как изменить универсальный тип поля в подклассе? - PullRequest
0 голосов
/ 08 февраля 2019

У меня есть класс, A:

import java.util.ArrayList;
import java.util.List;

public class A {

    protected List<Object> objects;

    public A() {
        objects = new ArrayList<Object>();
    }

    public A(List<Object> objects) {
        this.objects = objects;
    }

    public List<Object> getObjects() {
        return objects;
    }

    public void addObject(Object o) {
        getObjects().add(o);
    }

}

Я хочу разделить его на класс с именем B, но изменив поле objects на List<Number>:

import java.util.List;

public class B extends A {

    @Override
    public List<Number> getObjects() {
        return (List<Number>) super.getObjects();
    }

    @Override
    public void addObject(Object o) {
        getObjects().add((Number) o);
    }
}

Это, однако, приводит к ошибке компилятора.Как я могу сделать это правильно?

Ответы [ 2 ]

0 голосов
/ 08 февраля 2019

Это ключевая тонкость работы дженериков в Java.Если тип T расширяет тип U, мы ожидаем, что List<T> расширяет List<U>.Но это не так.Поэтому, когда вы пытаетесь вернуть List<Number> в качестве подтипа List<Object>, компилятор жалуется.

Обратите внимание: жалоба не связана с приведением, которое вы выполняете внутри тела функции, что само по себе опасно.Речь идет о типе возвращаемого значения функции, которая претендует на переопределение.В причудливых терминах тип возвращаемого значения не является «ковариантным» с методом в суперклассе: https://www.tutorialspoint.com/Covariant-return-types-in-Java.

Так что это объясняет «почему» за ошибкой компилятора.

Что касается «как это сделать правильно.Не видя варианта использования, я должен сказать, что Ответ Брета С. - это то, как я бы это сделал.

0 голосов
/ 08 февраля 2019

Введите параметр типа в свой класс A ...

import java.util.ArrayList;
import java.util.List;

public class A<T> {

    protected List<T> objects;

    public A() {
        objects = new ArrayList<T>();
    }

    public A(List<T> objects) {
        this.objects = objects;
    }

    public List<T> getObjects() {
        return objects;
    }

    public void addObject(T o) {
        getObjects().add(o);
    }

}

Теперь B фактически не нужно переопределять методы ...

public class B extends A<Number> {

}
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...