Java泛型构造函数

1.概述

咱们以前讨论过Java Generics的基础知识。在本文中,咱们将了解Java中的通用构造函数。 泛型构造函数是至少须要有一个泛型类型参数的构造函数。咱们将看到泛型构造函数并不都是在泛型类中出现的,并且并不是全部泛型类中的构造函数都必须是泛型。java

2.非泛型类

首先,先写一个简单的类:Entry,它不是泛型类:web

public class Entry { private String data; private int rank; } 复制代码

在这个类中,咱们将添加两个构造函数:一个带有两个参数的基本构造函数和一个通用构造函数。ide

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是受限制的,应该实现RankableSerializable接口。this

如今,让咱们看看Rankable接口,下面是其中一个方法:spa

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()); } 复制代码

请注意,咱们应该可以将可选参数类型(Product示例)转换为GenericEntry类型(Serializable示例)。

7.结束语

在本文中,咱们学习了如何在泛型和非泛型类中定义和使用泛型构造函数。

完整的源代码能够在GitHub获取(点击查看原文)。

原文连接:www.baeldung.com/java-generi…

做者:baeldung

译者:Emma

推荐关注公众号:锅外的大佬

每日推送国外优秀的技术翻译文章,励志帮助国内的开发者更好地成长!

 

相关文章
相关标签/搜索