首页 > Java > java教程 > 正文

java泛型容器Collection怎么用

WBOY
发布: 2023-04-19 18:10:42
转载
1016人浏览过

先简单来段例子:

<code>public void testGenerics() {<br/>    Collection<Number> numbers = new ArrayList<>();<br/>    numbers.add(1); // ok<br/>    numbers.add(0.1); // ok<br/><br/>    Collection<? extends Number> numbers2 = new ArrayList<>();<br/>    // don't work, you don't know which subtype 'numbers2' exactly contains<br/>    numbers2.add(1); // oops!<br/>}<br/></code>
登录后复制

这个例子其实有点反人类,估计大部分人(包括我)对这种转换的第一反应肯定是“当然是对的”(这就掉坑了),说下我的理解:

  • Collection<Number>:表示这个Collection里包含的都是Number类型的对象,可以是Integer/Long/Float,因为编译器可以判断obj instanceof Number == true;

  • Collection<? extends Number>:表示这个Collection是Number类型的“某个子类型”的Collection实例,可以是Collection<Integer>/Collection<Long>,所以调用numbers2.add(1)是不行的,因为编译器不知道这个numbers2包含的元素到底是Number的哪个子类型,编译器无法判断obj instanceof UnknownType的结果;

  • Collection<E>,这个E类型是“一个”具体的类型,而不能是表示某个parent的多种子类型的占位符;

    立即学习Java免费学习笔记(深入)”;

再来个例子:

<code>public void testGenerics() {<br/>    Collection<Number> numbers = new ArrayList<>();<br/>    Collection<Integer> integers = new ArrayList<>();<br/>    Collection<? extends Number> numbers2 = new ArrayList<>();<br/>    <br/>    numbers2 = integers; // ok<br/>    numbers2 = numbers; // ok<br/>    <br/>    // don't work, Collection<Number> != Collection<Integer><br/>    numbers = integers; // oops!<br/>}<br/></code>
登录后复制

Integer明明继承了Number,那为什么

  • Collection<Number> == Collection<Integer>

不成立呢,我们再来看个例子:

<code>public void testGenerics() {<br/>    Collection<Integer> profits = new ArrayList<>();<br/>    <br/>    insertSomething(profits); // line 1<br/>    <br/>    Integer profit = profits.iterator().next(); // oops! crash<br/>}<br/><br/>private void insertSomething(Collection<Number> numbers) {<br/>    numbers.add(Long.MAX_VALUE);<br/>}<br/></code>
登录后复制

如果line 1成立,那么接下去获取利润将会得到个负数,后续的一系列计算都会发声异常,如果代码不够健壮甚至可能会抛出一些意料之外的RuntimeException,导致方法不正常结束甚至程序crash。

所以一句话,Collection<Number> != Collection<Integer>是为了运行期的安全,将可能发生的类型转换异常在编译期就解决掉。

无阶未来模型擂台/AI 应用平台
无阶未来模型擂台/AI 应用平台

无阶未来模型擂台/AI 应用平台,一站式模型+应用平台

无阶未来模型擂台/AI 应用平台 35
查看详情 无阶未来模型擂台/AI 应用平台

现在再来说说Collection<Object>与Collection<?>,又是很多人(包括我)第一反应肯定是“Object是所有java对象的公共父类,所以Collection<Object>可以表示任意类型的集合”,来看个例子:

<code>public void testGenerics2() {<br/>    Collection<Integer> integers = new ArrayList<>();<br/><br/>    Collection<?> objects2 = integers; // ok<br/>    // don't work, which type of 'objects2' contains is uncertain<br/>    objects2.add(1); // oops!<br/>    <br/>    Collection<Object> objects = integers; // oops!<br/>}<br/></code>
登录后复制
  • Collection<?>表示的范围比Collection<Object>大;

  • 无法调用objects2.add(1)是因为编译器无法精确推断objects2到底是哪种数据类型的容器,可能会产生运行时的类型转换异常;

  • 表示任意数据类型集合的正确写法是Collection<?>;

  • Collection<Object>不能表示任意类型的集合。

为什么Collection<Object>不是表示任意类型的集合呢,其实也是编译器认为这里有类型转换错误的风险:

<code>public void testGenerics() {<br/>    Collection<Integer> integers = new ArrayList<>();<br/><br/>    Collection<Object> objects = integers; // oops!<br/>    // don't work, which type of 'objects2' contains is uncertain<br/>    objects.add("1");<br/><br/>    Integer one = objects.iterator().next(); // oops! crash<br/>}<br/></code>
登录后复制
  • Collection<Object>是可以往容器add数据的,因为Object是所有对象的父类,是已知类型,可以用obj instanceof Object判断;

  • Collection<?>不能往容器放数据,因为? (UnknownType)是未知类型,无法判断obj instanceof UnknownType的结果;

  • ?是表示未知类型,Object表示的是已知类型;

  • 如果Collection<Object>表示任意类型,按照墨菲定律(可能会发生的事必然会发生),那么上面例子中的crash是必然会发生的。。(又是一个线上故障)

以上就是java泛型容器Collection怎么用的详细内容,更多请关注php中文网其它相关文章!

相关标签:
java速学教程(入门到精通)
java速学教程(入门到精通)

java怎么学习?java怎么入门?java在哪学?java怎么学才快?不用担心,这里为大家提供了java速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!

下载
来源:亿速云网
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
最新问题
开源免费商场系统广告
热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板
关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新 English
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送
PHP中文网APP
随时随地碎片化学习

Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号