美文网首页
当类的泛型相关时,如何在两个泛型类之间创建类似子类型的关系

当类的泛型相关时,如何在两个泛型类之间创建类似子类型的关系

作者: 阿Q说代码 | 来源:发表于2022-01-17 09:16 被阅读0次

哈喽大家好,我是阿Q!

事情是这个样子的......

image

对话中的截图如下:

image

看了阿Q的解释,你是否也和“马小跳”一样存在疑问呢?请往👇看

我们都知道在java中,只要是类型兼容,就可以将一种类型的对象分配给另一种类型的对象。比如可以将一个Integer类型的对象分配给Object类型的对象,因为ObjectInteger的超类。

Object someObject = new Object();
Integer someInteger = new Integer(10);
someObject = someInteger; //OK

在面向对象中,我们把它称之为 is a 的关系。因为IntegerObject的一种子类,所以允许被赋值。

又因为Integer也是Number的一种子类,所以下边的代码也是有效的:

public void someMethod(Number n) { /* ... */ }

someMethod(new Integer(10));   // OK
someMethod(new Double(10.1));   // OK

当然泛型也是如此,在执行泛型类型调用时,将Number作为其类型参数传递,如果参数是Number的子类型,则允许任何后续的add调用:

Box<Number> box = new Box<Number>();
box.add(new Integer(10));   // OK
box.add(new Double(10.1));  // OK

现在我们来看以下代码:

public void boxTest(Box<Number> n) { /* ... */ }

该方法接收什么类型的参数呢?

通过该方法,大家肯定知道它的参数类型为Box<Number>,但是大家思考一个问题:你认为Box<Integer>Box<Double>类型的参数可以传入吗?

答案是否定的。

image

尽管IntegerNumber的子类型,但Box<Integer>Box<Double>不是Box<Number>的子类,它俩的父类对象是Object文首的对话表达的就是这个意思。

那么问题来了,当类的泛型相关时,如何在两个泛型类之间创建类似子类型的关系呢?例如如何让Box<Integer>Box<Double>变得与Box<Number>有关呢?

为了搞懂这个问题,我们先来了解一下同一类型的对象是如何实现子类型化的吧。

image

通过分析源码我们可以发现:ArrayList<E> 实现了 List<E>List<E> 继承了Collection<E>,所以ArrayList<String>List<String>的子类型, List<String>Collection<String>的子类型。因此当我们在传递参数时,ArrayList<String>类型的是可以给List<E>或者Collection<E>传递的。

只要不改变类型参数,类型之间的子类型关系就会保留。

如果我们想要定义我们自己的列表接口PayloadList,使得泛型类型P的可选值与每个元素相关联,可以定义如下:

interface PayloadList<E,P> extends List<E> {
  void setPayload(int index, P val);
  ...
}
image

PayloadList<String,String>PayloadList<String,Integer>PayloadList<String,Exception>都是List<String>的子类型。

小结:可以通过继承泛型类或者实现接口来对其进行子类型化。

搞懂了子类型化的问题,我们回到“如何在两个泛型类之间创建类似子类型的关系“的问题。

泛型类或者接口并不会仅仅因为它们的类型之间有关系而变得相关,如果要达到相关,我们可以使用通配符来创建泛型类或接口之间的关系。

image

Box<Integer>Box<Number>的父类对象其实是Box<?>

为了在这些类之间创建关系,以便代码可以通过Box<Integer>访问Box<Number>的方法,可以使用上限通配符:

Box<? extends Integer> initBox = new Box<>();
Box<? extends Number> numberBox = initBox;
// OK. Box<? extends Integer> is a subtype of Box<? extends Number>

因为IntegerNumber的子类型,numberBox的泛型是Number对象子类,所以在intBoxnumberBox之间存在关系。

image

图为用上限下限通配符声明的几个类之间的关系。

所以,“马小跳”的问题你会了吗?还不会的话来技术群交流吧!

相关文章

  • 当类的泛型相关时,如何在两个泛型类之间创建类似子类型的关系

    哈喽大家好,我是阿Q! 事情是这个样子的...... 对话中的截图如下: 看了阿Q的解释,你是否也和“马小跳”一样...

  • Kotlin学习(9)泛型

    9.1 泛型类型参数 ​ 泛型可以让你将类型定义为类型参数。当这样的一个类的实例被创建时,类型参数就被指定的类...

  • 泛型类

    注意:泛型类型必须是引用类型 泛型类,把泛型定义在类上 格式:public class 类名<泛型类型1,…> 把...

  • 【Swift】泛型常见使用

    1、Swift泛型4种 泛型函数泛型类型泛型协议泛型约束 2、泛型约束3种 继承约束:泛型类型 必须 是某个类的子...

  • typescript入门-泛型

    基础定义 泛型接口 泛型类 泛型约束 使用类类型

  • Kotlin 泛型

    Kotlin 支持泛型, 语法和 Java 类似。例如,泛型类: 泛型函数: 类型变异 Java 的泛型中,最难理...

  • 泛型

    泛型的本质是参数化类型,即将操作数的数据类型指定为一个参数。当这种参数类型用在类中时,就称该类为泛型类。泛型类的格...

  • C#入门经典(第6版)阅读笔记(第三篇)(泛型)

    目录#####第十二章 泛型****#####第十二章 泛型个人理解,泛型就是在创建时以改变类的类型参数和函数的类...

  • 34. 泛型

    泛型即参数化类型。将类型参数化,可以用在类、接口、方法上。泛型类 创建类的实例时,要指定类型的参数。或者赋值让编译...

  • 泛型(持续更新中......)

    一、泛型类和泛型接口 1.如果定义了泛型类,实例化没有指明类的泛型,则认为此泛型类型为Object类型。2.要求:...

网友评论

      本文标题:当类的泛型相关时,如何在两个泛型类之间创建类似子类型的关系

      本文链接:https://www.haomeiwen.com/subject/rmfchrtx.html