经验首页 前端设计 程序设计 Java相关 移动开发 数据库/运维 软件/图像 大数据/云计算 其他经验
当前位置:技术经验 » Java相关 » Spring Boot » 查看文章
Spring Boot实现STOMP协议的WebSocket的方法步骤
来源:jb51  时间:2019/5/31 8:35:54  对本文有异议

1.概述

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

2.非泛型类

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

  1. public class Entry {
  2. private String data;
  3. private int rank;
  4. }
  5.  

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

2.1 基本构造器

Entry第一个构造函数:带有两个参数的简单构造函数:

  1. public Entry(String data, int rank) {
  2. this.data = data;
  3. this.rank = rank;
  4. }
  5.  

现在,让我们使用这个基本构造函数来创建一个Entry对象

  1. @Test
  2. public void givenNonGenericConstructor_whenCreateNonGenericEntry_thenOK() {
  3. Entry entry = new Entry("sample", 1);
  4.  
  5. assertEquals("sample", entry.getData());
  6. assertEquals(1, entry.getRank());
  7. }
  8.  

2.2 泛型构造器

接下来,第二个构造器是泛型构造器:

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

虽然Entry类不是通用的,但它有一个参数为E的泛型构造函数。

泛型类型E是受限制的,应该实现Rankable和Serializable接口。

现在,让我们看看Rankable接口,下面是其中一个方法:

  1. public interface Rankable {
  2. public int getRank();
  3. }
  4.  

假设我们有一个实现Rankable接口的类——Product

  1. public class Product implements Rankable, Serializable {
  2. private String name;
  3. private double price;
  4. private int sales;
  5.  
  6. public Product(String name, double price) {
  7. this.name = name;
  8. this.price = price;
  9. }
  10.  
  11. @Override
  12. public int getRank() {
  13. return sales;
  14. }
  15. }
  16.  

然后我们可以使用泛型构造函数和Product创建Entry对象:

  1. @Test
  2. public void givenGenericConstructor_whenCreateNonGenericEntry_thenOK() {
  3. Product product = new Product("milk", 2.5);
  4. product.setSales(30);
  5.  
  6. Entry entry = new Entry(product);
  7.  
  8. assertEquals(product.toString(), entry.getData());
  9. assertEquals(30, entry.getRank());
  10. }
  11.  

3.泛型类

接下来,我们看一下泛型类:GenericEntry

  1. public class GenericEntry<T> {
  2. private T data;
  3. private int rank;
  4. }
  5.  

我们将在此类中添加与上一节相同的两种类型的构造函数。

3.1 基础构造器

首先,让我们为GenericEntry类编写一个简单的非泛型构造函数:

  1. public GenericEntry(int rank) {
  2. this.rank = rank;
  3. }
  4.  

尽管GenericEntry是泛型类,但这是一个简单的,没有任何参数的构造函数。

现在,我们可以使用此构造函数来创建GenericEntry:

  1. @Test
  2. public void givenNonGenericConstructor_whenCreateGenericEntry_thenOK() {
  3. GenericEntry<String> entry = new GenericEntry<String>(1);
  4.  
  5. assertNull(entry.getData());
  6. assertEquals(1, entry.getRank());
  7. }
  8.  

3.2 泛型构造器

接下来,在类中添加第二个构造函数:

  1. public GenericEntry(T data, int rank) {
  2. this.data = data;
  3. this.rank = rank;
  4. }
  5.  

这是一个泛型构造函数,它有一个泛型类型T的数据参数。注意,我们不需要在构造函数声明中添加,因为它是隐含的。

现在,让我们测试一下通用构造函数:

  1. @Test
  2. public void givenGenericConstructor_whenCreateGenericEntry_thenOK() {
  3. GenericEntry<String> entry = new GenericEntry<String>("sample", 1);
  4.  
  5. assertEquals("sample", entry.getData());
  6. assertEquals(1, entry.getRank());
  7. }
  8.  

4.不同类型的泛型构造函数

在泛型类中,还有一个构造函数,其泛型类型与类的泛型类型不同:

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

GenericEntry构造函数有类型为E的参数,该参数与T类型不同。让我们看看它的实际效果:

  1. @Test
  2. public void givenGenericConstructorWithDifferentType_whenCreateGenericEntry_thenOK() {
  3. Product product = new Product("milk", 2.5);
  4. product.setSales(30);
  5.  
  6. GenericEntry<Serializable> entry = new GenericEntry<Serializable>(product);
  7.  
  8. assertEquals(product, entry.getData());
  9. assertEquals(30, entry.getRank());
  10. }

注意:在示例中,我们使用Product(E)创建Serializable(T)类型的GenericEntry,只有当类型E的参数可以转换为T时,我们才能使用此构造函数。

5.多种泛类型

接下来,我们有两个泛型类型参数的泛型类MapEntry:

  1. public class MapEntry<K, V> {
  2. private K key;
  3. private V value;
  4.  
  5. public MapEntry(K key, V value) {
  6. this.key = key;
  7. this.value = value;
  8. }
  9. }
  10.  

MapEntry有一个两个参数的泛型构造函数,每个参数都是不同的类型。让我们用一个简单的单元测试测试一下:

  1. @Test
  2. public void givenGenericConstructor_whenCreateGenericEntryWithTwoTypes_thenOK() {
  3. MapEntry<String,Integer> entry = new MapEntry<String,Integer>("sample", 1);
  4.  
  5. assertEquals("sample", entry.getKey());
  6. assertEquals(1, entry.getValue().intValue());
  7. }
  8.  

6.通配符

最后,我们可以在泛型构造函数中使用通配符:

  1. public GenericEntry(Optional<? extends Rankable> optional) {
  2. if (optional.isPresent()) {
  3. this.data = (T) optional.get();
  4. this.rank = optional.get().getRank();
  5. }
  6. }
  7.  

在这儿,我们在GenericEntry构造函数中使用通配符来绑定Optional类型:

  1. @Test
  2. public void givenGenericConstructorWithWildCard_whenCreateGenericEntry_thenOK() {
  3. Product product = new Product("milk", 2.5);
  4. product.setSales(30);
  5. Optional<Product> optional = Optional.of(product);
  6.  
  7. GenericEntry<Serializable> entry = new GenericEntry<Serializable>(optional);
  8.  
  9. assertEquals(product, entry.getData());
  10. assertEquals(30, entry.getRank());
  11. }
  12.  

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

7.结束语

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

原文链接:https://www.baeldung.com/java-generic-constructors

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持w3xue。

 友情链接:直通硅谷  点职佳  北美留学生论坛

本站QQ群:前端 618073944 | Java 606181507 | Python 626812652 | C/C++ 612253063 | 微信 634508462 | 苹果 692586424 | C#/.net 182808419 | PHP 305140648 | 运维 608723728

W3xue 的所有内容仅供测试,对任何法律问题及风险不承担任何责任。通过使用本站内容随之而来的风险与本站无关。
关于我们  |  意见建议  |  捐助我们  |  报错有奖  |  广告合作、友情链接(目前9元/月)请联系QQ:27243702 沸活量
皖ICP备17017327号-2 皖公网安备34020702000426号