Java 泛型中的通配符详解

news2025/1/11 23:01:44

目录

1、如何定义和使用上界通配符?

2、如何定义和使用无界通配符?

3、如何定义和使用下界通配符?

4、如何使用通配符定义泛型类或接口之间的子类型关系?

5、通配符的捕获和辅助方法

6、通配符使用指南


        在泛型代码中,问号(?)称为通配符,用来表示未知类型。通配符可以在多种情况下使用:如作为参数、字段或局部变量的类型;有时也可以作为返回类型。另外,通配符永远不会用作调用泛型方法、创建泛型类或超类型实例的类型参数。

        为什么使用通配符?

        在 Java 中,类和数组之间的对象关系是可以继承的,比如 Dog extends Animal,那么 Animal[] 与 Dog[] 就是兼容的。但是集合之间却不存在这种关系,也就是说 List<Animal> 不是List<Dog> 的父类,他们之间没有任何关系。那么为了建立两个集合之间的联系,就需要用到通配符。// 只是其中一部分原因

1、如何定义和使用上界通配符?

        可以使用上界通配符来放宽对变量的限制。例如,你想编写一个方法,该方法适用于List<Integer>, List<Double> 和 List<Number>;就可以通过使用上界通配符来实现这一点。

        声明一个上界通配符,需要使用通配符 ('?'),跟上 extends 关键字,然后再跟它的上界。比如,编写用于 Number 列表和 Number 子类型(如 Integer、Double 和 Float)的方法,可以指定 List<? extends Number>。List<? extends Number> 要比 List<Number> 对类型的限制更加宽松,因为 List<Number> 只匹配类型为 Number 的列表,而 List<? extends Number> 匹配类型为 Number 或其任何子类的列表。

        例如下边的程序代码:

public static void process(List<? extends Foo> list) { /* ... */ }

        上界通配符 <? extends Foo> 中的 Foo 匹配 Foo 类型和 Foo 的任何子类型。process() 方法可以访问类型为 Foo 的列表元素:

public static void process(List<? extends Foo> list) {
    for (Foo elem : list) {
        // ...
    }
}

        在 foreach 子句中,elem 变量迭代列表中的每个元素。在 elem 元素上可以使用 Foo 类中定义的任何方法

        如下,sumOfList() 方法返回列表中数字的和:

public static double sumOfList(List<? extends Number> list) {
    double s = 0.0;
    for (Number n : list)
        s += n.doubleValue();
    return s;
}

        下面的代码,使用一个 Integer 对象列表,输出 sum = 6.0:

List<Integer> li = Arrays.asList(1, 2, 3);
System.out.println("sum = " + sumOfList(li));

        Double 值列表可以使用相同的 sumOfList() 方法。下面的代码输出 sum = 7.0:

List<Double> ld = Arrays.asList(1.2, 2.3, 3.5);
System.out.println("sum = " + sumOfList(ld));

2、如何定义和使用无界通配符?

        无界通配符类型使用通配符(?)指定,例如 List<?>。在下边两种情况下,无界通配符是一种有用的方法:

  1. 编写一个方法,该方法可以使用 Object 类中提供的功能函数。
  2. 代码在泛型类中使用不依赖于类型参数的方法。例如,List.size 或 List.clear。事实上,Class<?> 之所以如此常用,是因为 Class<T> 中的大多数方法都不依赖于 T。

        例如以下 printList() 方法:

public static void printList(List<Object> list) {
    for (Object elem : list)
        System.out.println(elem + " ");
    System.out.println();
}

        printList() 的目标是用来打印任何类型的列表,但上述代码中它只能打印 Object 实例的列表;并不能打印 List<Integer>, List<String>, List<Double> 等,因为这些列表不是 List<Object> 的子类型。所以如果要编写一个通用的 printList() 方法,需要使用到 List<?>:

public static void printList(List<?> list) {
    for (Object elem: list)
        System.out.print(elem + " ");
    System.out.println();
}

        对于任何具体的类型 A,List<A> 都是 List<?> 的子类,可以使用 printList() 打印任何类型的列表:// 解决了之前 List<A> 和 List<B> 无任何关系的问题

List<Integer> li = Arrays.asList(1, 2, 3);
List<String>  ls = Arrays.asList("one", "two", "three");
printList(li);
printList(ls);

        注意:List<Object> 和 List<?> 是不一样的。可以将 Object 或 Object 的任何子类型插入到 List<Object> 中,但是只能将 null 插入到 List<?> 中。

3、如何定义和使用下界通配符?

        上界通配符将未知类型限制为特定类型的子类型,使用 extends 关键字表示。类似地,下界通配符将未知类型限制为特定类型的超类型,使用 super 关键字表示

        下界通配符使用通配符('?')表示,后面跟着 super 关键字,然后再跟着它的下界:<? super A>。

        注意:可以为通配符指定上界,也可以指定下界,但不能同时都指定。

        例如,编写一个将 Integer 对象放入列表的方法。为了最大限度地提高灵活性,该方法需要适用于 List<Integer>、List<Number> 和 List<Object> 等任何可以保存 Integer 值的对象。那么要编写处理 Integer 列表和 Integer 超类型列表的方法,就可以使用 List<? super Integer> 来指定。如以下代码将数字 1 到 10 添加到列表的末尾:

public static void addNumbers(List<? super Integer> list) {
    for (int i = 1; i <= 10; i++) {
        list.add(i);
    }
}

4、如何使用通配符定义泛型类或接口之间的子类型关系?

        泛型类或接口之间的关联并不取决于它们的类型之间是否存在关联。不过,我们可以使用通配符来创建泛型类或接口之间的关联关系

        给定以下两个非泛型类:

class A { /* ... */ }
class B extends A { /* ... */ }

        编写以下代码是合理的:

B b = new B();
A a = b;

        上边这个例子表明常规类的继承遵循子类型的规则:如果 B 继承 A,那么 B.calss 就是 A.calss 的子类型。但是这个规则并不适用于泛型类型:

List<B> lb = new ArrayList<>();
List<A> la = lb;   // compile-time error

        假设 Integer 是 Number 的子类型,那么 List<Integer> 和 List<Number> 之间的关系是什么呢?

        虽然 Integer 是 Number 的子类型,但 List<Integer> 并不是 List<Number> 的子类型,所以,这两种类型并不相关。事实上,List<Number> 和 List<Integer> 的公共父类是 List<?>。

        为了在这些类之间创建关系,让代码可以通过 List<Integer> 的元素访问 Number 的方法,可以使用一个上界通配符:

List<? extends Integer> intList = new ArrayList<>();
List<? extends Number>  numList = intList;  // OK. List<? extends Integer> is a subtype of List<? extends Number>

        因为 Integer 是 Number 的子类型,而 numList 是 Number 对象的列表,所以 intList (Integer对象的列表)和 numList 之间存在关联关系。下图显示了用上下界通配符声明的几个 List 类之间的关系。// 使用通配符可以定义两个类型之间的关系

5、通配符的捕获和辅助方法

        在某些情况下,编译器会自动推断通配符的类型。例如,列表可以定义为 List<?>,但是,当计算表达式时,编译器会从代码中推断出特定的类型,这种场景称为通配符捕获。

        大多数情况下,都不需要担心通配符的捕获,除非看到包含短语 “capture of” 的错误消息。

        如下,WildcardError 示例会在编译时会产生一个捕获错误:

import java.util.List;

public class WildcardError {

    void foo(List<?> i) {
        i.set(0, i.get(0));
    }
}

        在本例中,编译器将 i 输入形参处理为 Object 类型。当 foo 方法调用 List.set(int, E) 时,编译器无法确认插入到列表中的对象类型,所以会产生错误。当发生这种类型的错误时,通常意味着编译器认为给变量分配了错误的类型。将泛型添加到 Java 语言中就是出于这个原因——在编译时加强类型安全。

        当使用 Oracle 的 JDK 7 javac 实现编译时,WildcardError 示例会生成以下错误:

WildcardError.java:6: error: method set in interface List<E> cannot be applied to given types;
    i.set(0, i.get(0));
     ^
  required: int,CAP#1
  found: int,Object
  reason: actual argument Object cannot be converted to CAP#1 by method invocation conversion
  where E is a type-variable:
    E extends Object declared in interface List
  where CAP#1 is a fresh type-variable:
    CAP#1 extends Object from capture of ?
1 error

        在本例中,代码试图执行安全操作,那么如何解决编译器错误呢?可以通过编写一个捕获通配符的私有 helper 方法来修复它。如 WildcardFixed 所示,通过创建私有辅助方法 fooHelper 来解决这个问题:

public class WildcardFixed {

    void foo(List<?> i) {
        fooHelper(i);
    }

    // 创建辅助方法,以便可以通过类型推断捕获通配符
    private <T> void fooHelper(List<T> l) {
        l.set(0, l.get(0));
    }
}

        由于使用了 helper 方法,编译器使用推断来确定调用中的 T 是 CAP#1,即捕获变量。示例现在编译成功。

6、通配符使用指南

        在使用泛型进行编程时,让人比较困惑的是需要确定何时使用上界通配符,何时使用下界通配符。本节提供了一些设计代码时要遵循的指导原则。// 以下是来自官方的指导原则

        为了方便理解,可以将变量看作以下两种形式:

  • “in” 变量:“in” 变量为代码提供数据。比如一个复制方法有两个参数:copy(src, dest)。src 参数提供了需要复制的数据,因此它是 “in” 形参。
  • “out” 变量:“out” 变量用来保存输出数据,便于该数据在其他地方使用。在 copy(src, dest) 方法中, dest 参数用来接收数据,因此它是 "out" 形参。

        当然,有些变量同时用于 “输入” 和 “输出” 目的,下边指南中也提到了这种情况。

        在决定是否使用通配符以及使用什么类型的通配符时,可以使用 “in” 和 “out” 原则。以下列表提供了需要遵循的指导方针:

  1. 对于 “in” 变量,可以定义一个上界通配符,使用 extends 关键字。
  2. 对于“out”变量,可以定义一个下界通配符,使用 super 关键字。
  3. 如果 “in” 变量可以使用 Object.calss 中定义的方法对其进行访问,可以使用无界通配符。
  4. 如果变量同时用于 “输入” 和 “输出”的情况下,不要使用通配符。// 限定唯一的类型

        以上这些准则不适用于方法的返回类型。应该避免使用通配符作为方法的返回类型,因为它会迫使使用代码的程序员去处理通配符。

        List<? extends ...> 可以认为它是只读的,但并不能进行严格的保证,假设有以下两个类:

class NaturalNumber {

    private int i;

    public NaturalNumber(int i) { this.i = i; }
    // ...
}

class EvenNumber extends NaturalNumber {

    public EvenNumber(int i) { super(i); }
    // ...
}

        考虑下面的代码:

List<EvenNumber> le = new ArrayList<>();
List<? extends NaturalNumber> ln = le;
ln.add(new NaturalNumber(35));  // 编译错误

        因为 List<EvenNumber> 是 List<? extends NaturalNumber> 的子类型,所以可以将 le 赋值给 ln。但是却不能用 ln 把一个自然数加到一个偶数列表中。在该列表中只可以进行以下操作:

  1. 添加 null 元素。
  2. 调用 clear() 方法。
  3. 获取迭代器并调用 remove() 方法。
  4. 捕获通配符并写入从列表中读取的元素。

        所以,从严格意义上来讲,List<? extends NaturalNumber> 并不是只读的,但是却可以认为它是只读的,因为不能在列表中存储新的元素或更改现有元素

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/149626.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

C++ txt文本文件处理系统(c++学习小例子)

C++ txt文本文件处理系统(c++学习小例子) 一、界面示例二、 要求2.1 数据格式2.2 实现功能三、代码3.1 classfi.h3.2 classfi.cpp3.3 main.cpp四、 使用说明一、界面示例 二、 要求 2.1 数据格式 现有DEM数据,其格式为DEM_data.txt,可在文章末尾下载。文本存储格式如下: …

用javascript分类刷leetcode19.数组(图文视频讲解)

数组操作的时间复杂度 Access&#xff1a;O(1) Search&#xff1a;O(n) Insert&#xff1a; 平均O(n)&#xff0c;最好的情况下O(1)&#xff0c;也就是在数组尾部插入O(1)&#xff0c;最坏的情况下O(n) Delete&#xff1b;平均O(n)&#xff0c;最好的情况下O(1)&#xff0c;…

力扣刷题记录——367. 有效的完全平方数、383. 赎金信、387. 字符串中的第一个唯一字符、389. 找不同

本专栏主要记录力扣的刷题记录&#xff0c;备战蓝桥杯&#xff0c;供复盘和优化算法使用&#xff0c;也希望给大家带来帮助&#xff0c;博主是算法小白&#xff0c;希望各位大佬不要见笑&#xff0c;今天要分享的是——《367. 有效的完全平方数、383. 赎金信、387. 字符串中的第…

LightDB单机安装

LightDB单机安装 LightDB官网&#xff1a;https://www.hs.net/lightdb 下载安装包&#xff1a;lightdb-x-13.8-22.3-7953-el7.x86_64.zip 前置准备 防火墙配置(选择一种操作) firewall防火墙 firewall-cmd --permanent --add-port5432/tcp firewall-cmd --permanent --add-p…

(深度学习快速入门)第三章第三节2:深度学习必备组件之损失函数和激活函数

文章目录一&#xff1a;损失函数&#xff08;1&#xff09;均方误差损失&#xff08;MSE&#xff09;&#xff08;2&#xff09;交叉熵损失&#xff08;Cross Entropy&#xff09;二&#xff1a;激活函数&#xff08;1&#xff09;tanh&#xff08;2&#xff09;ReLU&#xff0…

SpringBoot数据响应与内容协商

目录 数据响应与内容协商 1、响应JSON 1.1、jackson.jarResponseBody 1.2、SpringMVC到底支持哪些返回值 2、内容协商 1、引入xml依赖 2、postman分别测试返回json和xml 3、开启浏览器参数方式内容协商功能 数据响应与内容协商 1、响应JSON 1.1、jackson.jarResponseBo…

区块链北大肖老师学习笔记6

第七节 比特币的挖矿难度调整 H(block header) < target 目标(target)预值越小&#xff0c;挖矿的难度越大。调整挖矿的难度就是调整目标空间在整个输出空间中所占的比例。 比特币用的哈希算法是SHA-256&#xff0c;这个产生的哈希值是256位。所以整个输出空间是2的256次…

[JavaEE]synchronized 与 死锁

专栏简介: JavaEE从入门到进阶 题目来源: leetcode,牛客,剑指offer. 创作目标: 记录学习JavaEE学习历程 希望在提升自己的同时,帮助他人,,与大家一起共同进步,互相成长. 学历代表过去,能力代表现在,学习能力代表未来! 目录 1.synchronized 的特性 2. synchronized 使用示例:…

new做了什么

function structure (name, age) {this.name namethis.age age}// 给构造函数--prototype加上一个方法structure.prototype.sayName function () {console.log(this.name, 调用打印);return this.name}structure.one 5const person new structure(张龙, 188)// 打印构造函…

【基础算法】前缀和 与 差分

前缀和 用来求解一段区间&#xff08;一维&#xff09;的总和 或者一块矩形区域&#xff08;二维&#xff09;的总和 一维前缀和 原数组a[N]&#xff0c;前缀和数组s[N]// ---读入数组a[N] // ---// 处理前缀和数组 s[N] s[0] 0; //定义在全局变量&#xff0c;不用写这一句 f…

SOA 和微服务有何区别?

玩过 Dubbo 的小伙伴应该都有听说过一个概念叫做 SOA&#xff0c;每当我们说起微服务的时候&#xff0c;很多人就会去纠结这和 SOA 有啥关系呀&#xff1f;感觉换汤不换药呀。 今天松哥来稍微和小伙伴们讨论下这个话题&#xff0c;我们一起来看看 SOA 和微服务到底有何异同。 …

c语言进阶(3)——指针进阶笔试题详解

1.指针和数组笔试题解析 关键&#xff1a;数组名在两种情况下是指整个数组&#xff1a; &#xff08;1&#xff09;sizeof&#xff08;数组名&#xff09;&#xff08;2&#xff09;&数组名 其它的情况下&#xff0c;都是代表数组的首元素地址。 例题 1 &#xff1a;一维…

【算法面试】算法在面试中考察的是什么(金三银四面试专栏启动)

&#x1f4eb;作者简介&#xff1a;小明java问道之路&#xff0c;专注于研究 Java/Liunx内核/C及汇编/计算机底层原理/源码&#xff0c;就职于大型金融公司后端高级工程师&#xff0c;擅长交易领域的高安全/可用/并发/性能的架构设计与演进、系统优化与稳定性建设。 &#x1f4…

Kafka消息中间件

Kafka消息中间件 同时市场上也发展处ActiveMq、RabbitMQ、Kafka、RocketMQ、Pulsar等众多优秀的框架&#xff1b;在大数据领域中Kafka目前是使用较多的框架。Kafka作为内部消息通知的框架&#xff0c;可以适应项目中大数据量的高吞吐、实时流计算等功能实现。 分布式消息中间…

【自学C++】C++整型

C整型 C整型教程 C 的整型用来存放整数 类型 的数字&#xff0c;即不可以带小数&#xff0c;C 整型可以分为短整型 short&#xff0c;整型 int&#xff0c;长整型 long 和 long long 类型。 C整型取值范围 数据类型取值范围字节数short-32768 ~ 327672int-2147483648 ~ 214…

小米万兆路由器里的Docker安装drawio

小米2022年12月份发布了万兆路由器&#xff0c;里面可以使用Docker。 今天尝试在小米的万兆路由器里安装drawio 20.8.3。 准备工作 请参考https://engchina.blog.csdn.net/article/details/128515422的准备工作。 查看Docker Hub镜像信息 访问https://hub.docker.com/r/jgr…

axios系列之并发

文章の目录一、axios.all(iterable)二、axios.spread(callback)写在最后处理并发请求的助手函数 一、axios.all(iterable) function a() {return axios.get("http://localhost/a"); } function b() {return axios.get("http://localhost/b"); }axios.all…

【Linux 进程控制】进程创建、进程终止、进程等待、进程替换

1.进程创建&#xff08;fork&#xff09;#include<iostream> #include<unistd.h> using std::cout; using std::endl;int main() {if(fork()0){cout<<"child:"<<"I am child"<<endl;}else{cout<<"parent:"&…

用555定时器接成的施密特触发器电路/滞回电压比较器中,用什么方法能调节回差电压的大小?包含工作原理与应用

一、简答:电源电压或外接控制电压改变时&#xff0c;可以改变回差电压的大小。二、施密特触发器电路工作原理&#xff1a;滞回电压比较器&#xff0c;又名施密特触发器&#xff0c;有两个稳定状态&#xff0c;与一般触发器不同的是&#xff0c;施密特触发器采用电位触发方式&am…

【免杀】通用shellcode原理及思路——FS段寄存器获取kernel32.dll基址逻辑、根据函数名进行查找逻辑、双指针循环遍历获取函数名称

通用shellcode通用shellcode思路FS段寄存器获取kernel32.dll基址逻辑根据函数名进行查找逻辑双指针循环遍历获取函数字符串总结通用shellcode思路 1、保存相关字符串 user32.dll、LoadLibraryA、GetProcAddress、MessageBoxA、hello 51hook 2、通过fs寄存器获取kernel32.dll…