欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

不同Java泛型構(gòu)造函數(shù)的詳解

 更新時間:2019年06月10日 10:55:18   作者:鍋外的大佬  
這篇文章主要介紹了不同Java泛型構(gòu)造函數(shù)的詳解,因為對象是應(yīng)用類型,對象賦值是指向同一個對象,所以如果需要保存對象某個時刻的狀態(tài),就需要構(gòu)造函數(shù)來new一個新的對象。下面我們來詳細(xì)了解一下吧

1.概述

我們之前討論過Java Generics的基礎(chǔ)知識。在本文中,我們將了解Java中的通用構(gòu)造函數(shù)。
泛型構(gòu)造函數(shù)是至少需要有一個泛型類型參數(shù)的構(gòu)造函數(shù)。我們將看到泛型構(gòu)造函數(shù)并不都是在泛型類中出現(xiàn)的,而且并非所有泛型類中的構(gòu)造函數(shù)都必須是泛型。

2.非泛型類

首先,先寫一個簡單的類:Entry,它不是泛型類:

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

在這個類中,我們將添加兩個構(gòu)造函數(shù):一個帶有兩個參數(shù)的基本構(gòu)造函數(shù)和一個通用構(gòu)造函數(shù)。

2.1 基本構(gòu)造器

Entry第一個構(gòu)造函數(shù):帶有兩個參數(shù)的簡單構(gòu)造函數(shù):

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

現(xiàn)在,讓我們使用這個基本構(gòu)造函數(shù)來創(chuàng)建一個Entry對象

@Test
public void givenNonGenericConstructor_whenCreateNonGenericEntry_thenOK() {
Entry entry = new Entry("sample", 1);
assertEquals("sample", entry.getData());
assertEquals(1, entry.getRank());
}

2.2 泛型構(gòu)造器

接下來,第二個構(gòu)造器是泛型構(gòu)造器:

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

雖然Entry類不是通用的,但它有一個參數(shù)為E的泛型構(gòu)造函數(shù)。

泛型類型E是受限制的,應(yīng)該實現(xiàn)Rankable和Serializable接口。

現(xiàn)在,讓我們看看Rankable接口,下面是其中一個方法:

public interface Rankable {
public int getRank();
}

假設(shè)我們有一個實現(xiàn)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;
}
}

然后我們可以使用泛型構(gòu)造函數(shù)和Product創(chuàng)建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;
}

我們將在此類中添加與上一節(jié)相同的兩種類型的構(gòu)造函數(shù)。

3.1 基礎(chǔ)構(gòu)造器

首先,讓我們?yōu)镚enericEntry類編寫一個簡單的非泛型構(gòu)造函數(shù):

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

盡管GenericEntry是泛型類,但這是一個簡單的,沒有任何參數(shù)的構(gòu)造函數(shù)。
現(xiàn)在,我們可以使用此構(gòu)造函數(shù)來創(chuàng)建GenericEntry:

@Test
public void givenNonGenericConstructor_whenCreateGenericEntry_thenOK() {
GenericEntry<String> entry = new GenericEntry<String>(1);
assertNull(entry.getData());
assertEquals(1, entry.getRank());
}

3.2 泛型構(gòu)造器

接下來,在類中添加第二個構(gòu)造函數(shù):

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

這是一個泛型構(gòu)造函數(shù),它有一個泛型類型T的數(shù)據(jù)參數(shù)。注意,我們不需要在構(gòu)造函數(shù)聲明中添加,因為它是隱含的。
現(xiàn)在,讓我們測試一下通用構(gòu)造函數(shù):

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

4.不同類型的泛型構(gòu)造函數(shù)

在泛型類中,還有一個構(gòu)造函數(shù),其泛型類型與類的泛型類型不同:

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

GenericEntry構(gòu)造函數(shù)有類型為E的參數(shù),該參數(shù)與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)創(chuàng)建Serializable(T)類型的GenericEntry,只有當(dāng)類型E的參數(shù)可以轉(zhuǎn)換為T時,我們才能使用此構(gòu)造函數(shù)。

5.多種泛類型

接下來,我們有兩個泛型類型參數(shù)的泛型類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有一個兩個參數(shù)的泛型構(gòu)造函數(shù),每個參數(shù)都是不同的類型。讓我們用一個簡單的單元測試測試一下:

@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.通配符

最后,我們可以在泛型構(gòu)造函數(shù)中使用通配符:

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

在這兒,我們在GenericEntry構(gòu)造函數(shù)中使用通配符來綁定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());
}

請注意,我們應(yīng)該能夠?qū)⒖蛇x參數(shù)類型(Product示例)轉(zhuǎn)換為GenericEntry類型(Serializable示例)。

7.結(jié)束語

在本文中,我們學(xué)習(xí)了如何在泛型和非泛型類中定義和使用泛型構(gòu)造函數(shù)。

以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

最新評論