Универсальный конструктор Java

Java

1 Обзор

мы обсуждали раньшеJava Genericsбазовые знания. В этой статье мы узнаем об универсальных конструкторах в Java. Универсальный конструктор — это конструктор, для которого требуется хотя бы один параметр универсального типа. Мы увидим, что универсальные конструкторы не всегда появляются в универсальных классах, и не все конструкторы в универсальных классах должны быть универсальными.

2. Неуниверсальные классы

Во-первых, напишите простой класс: Entry, который не является универсальным классом:

public class Entry {
    private String data;
    private int rank;
}

В этом классе мы добавим два конструктора: базовый конструктор с двумя параметрами и универсальный конструктор.

2.1 Базовый конструктор

EntryПервый конструктор: Простой конструктор с двумя параметрами:

public Entry(String data, int rank) {
    this.data = data;
    this.rank = rank;
}

Теперь давайте воспользуемся этим базовым конструктором для созданияEntryобъект

@Test
public void givenNonGenericConstructor_whenCreateNonGenericEntry_thenOK() {
    Entry entry = new Entry("sample", 1);

    assertEquals("sample", entry.getData());
    assertEquals(1, entry.getRank());
}

2.2 Общие конструкторы

Затем второй конструктор является универсальным конструктором:

public <E extends Rankable & Serializable> Entry(E element) {
    this.data = element.toString();
    this.rank = element.getRank();
}

Несмотря на то чтоEntryКласс не является универсальным, но у него есть параметр какEОбщий конструктор.

универсальный типEограничено и должно быть реализованоRankableиSerializableинтерфейс.

Теперь давайте посмотримRankableинтерфейс, вот один из методов:

public interface Rankable {
    public int getRank();
}

Предположим, у нас есть реализацияRankableКласс интерфейса -Product

public class Product implements Rankable, Serializable {
    private String name;
    private double price;
    private int sales;

    public Product(String name, double price) {
        this.name = name;
        this.price = price;
    }

    @Override
    public int getRank() {
        return sales;
    }
}

Затем мы можем использовать общий конструктор иProductСоздайтеEntryОбъект:

@Test
public void givenGenericConstructor_whenCreateNonGenericEntry_thenOK() {
    Product product = new Product("milk", 2.5);
    product.setSales(30);

    Entry entry = new Entry(product);

    assertEquals(product.toString(), entry.getData());
    assertEquals(30, entry.getRank());
}

3. Общий класс

Далее, давайте посмотрим на общий класс:GenericEntry

public class GenericEntry<T> {
    private T data;
    private int rank;
}

Мы добавим в этот класс те же два типа конструкторов, что и в предыдущем разделе.

3.1 Базовый конструктор

Во-первых, давайтеGenericEntryКласс пишет простой необобщенный конструктор:

public GenericEntry(int rank) {
    this.rank = rank;
}

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

Теперь мы можем использовать этот конструктор для созданияGenericEntry:

@Test
public void givenNonGenericConstructor_whenCreateGenericEntry_thenOK() {
    GenericEntry<String> entry = new GenericEntry<String>(1);

    assertNull(entry.getData());
    assertEquals(1, entry.getRank());
}

3.2 Общие конструкторы

Затем добавьте в класс второй конструктор:

public GenericEntry(T data, int rank) {
    this.data = data;
    this.rank = rank;
}

Это универсальный конструктор, который принимает параметр данных универсального типа T. Обратите внимание, что нам не нужно добавлять объявление конструктора, поскольку это подразумевается.

Теперь давайте проверим генеральный конструктор:

@Test
public void givenGenericConstructor_whenCreateGenericEntry_thenOK() {
    GenericEntry<String> entry = new GenericEntry<String>("sample", 1);

    assertEquals("sample", entry.getData());
    assertEquals(1, entry.getRank());        
}

4 различных типа универсального конструктора

В универсальном классе также есть конструктор, чей универсальный тип отличается от универсального типа класса:

public <E extends Rankable & Serializable> GenericEntry(E element) {
    this.data = (T) element;
    this.rank = element.getRank();
}

GenericEntryКонструктор имеет типEпараметр, аналогичныйTразличные виды. Давайте посмотрим на это в действии:

@Test
public void givenGenericConstructorWithDifferentType_whenCreateGenericEntry_thenOK() {
    Product product = new Product("milk", 2.5);
    product.setSales(30);

    GenericEntry<Serializable> entry = new GenericEntry<Serializable>(product);

    assertEquals(product, entry.getData());
    assertEquals(30, entry.getRank());
}

Примечание: в примере мы используемProduct(E)СоздайтеSerializable(T)ТипGenericEntry, только когда типEПараметры могут быть преобразованы вT, мы можем использовать этот конструктор.

5. Несколько универсальных типов

Далее у нас есть универсальный класс с двумя параметрами универсального типа.MapEntry:

public class MapEntry<K, V> {
    private K key;
    private V value;

    public MapEntry(K key, V value) {
        this.key = key;
        this.value = value;
    }
}

MapEntryСуществует общий конструктор с двумя параметрами, каждый из другого типа. Давайте проверим его простым тестом на единицу:

@Test
public void givenGenericConstructor_whenCreateGenericEntryWithTwoTypes_thenOK() {
    MapEntry<String,Integer> entry = new MapEntry<String,Integer>("sample", 1);

    assertEquals("sample", entry.getKey());
    assertEquals(1, entry.getValue().intValue());        
}

6. Подстановочные знаки

Наконец, мы можем использовать подстановочные знаки в универсальных конструкторах:

public GenericEntry(Optional<? extends Rankable> optional) {
    if (optional.isPresent()) {
        this.data = (T) optional.get();
        this.rank = optional.get().getRank();
    }
}

мы здесьGenericEntryИспользуйте подстановочные знаки в конструкторах для привязкиOptionalтип:

@Test
public void givenGenericConstructorWithWildCard_whenCreateGenericEntry_thenOK() {
    Product product = new Product("milk", 2.5);
    product.setSales(30);
    Optional<Product> optional = Optional.of(product);

    GenericEntry<Serializable> entry = new GenericEntry<Serializable>(optional);

    assertEquals(product, entry.getData());
    assertEquals(30, entry.getRank());
}

Обратите внимание, что мы должны иметь возможность преобразовать необязательный тип параметра (пример продукта) вGenericEntryТип (сериализуемый пример).

7. Заключение

В этой статье мы узнали, как определять и использовать универсальные конструкторы в универсальных и неуниверсальных классах.

Полный исходный код можно найти по адресуGitHubПолучить (нажмите, чтобы просмотреть исходный текст).

Оригинальная ссылка:Привет.Поставь Al Lington.com/java-day...

Добавить Автора

Переводчик: Эмма

Рекомендуется обратить внимание на паблик аккаунт: большой парень вне банка

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