java泛型探索——介绍篇
1. 泛型出现前后代码对比
先来看看泛型出现前,代码是这么写的:
List words = new ArrayList();
words.add("Hello ");
words.add("worlds!");
String s = (String) words.get(0) + (String) words.get(1);
System.out.println(s.equals("Hello worlds!"));
而泛型出现后无需做显式转换:
words.add("Hello ");
words.add("worlds!");
String s = words.get(0) + words.get(1);
System.out.println(s.equals("Hello worlds!"));
从本质上来说,以上代码编译为字节码后是一样的,对JVM而言没有区别。
2. 为什么有泛型?
泛型是什么?
泛型允许程序员编写代码时使用一些以后才指定的类型,允许在定义类和接口的时候使用类型参数,java泛型的参数只可以代表类,不能代表个别对象。
泛型的引入可以解决之前的集合类框架在使用过程中通常会出现的运行时刻类型错误[1]。
interface Iterator<E> {
public boolean hasNext();
public E next();
public void remove();
...
}
泛型方法也可以哦:
class List {
public static<T> List<T> toList(T[] arr) {
List<T> list = new ArrayList<T>();
for (T elt : arr) list.add(elt);
return list;
}
}
为什么有泛型?
1) 泛型可以隐式地执行类型转换,并且永远不会失败(cast-iron guarantee[2]):
List<String> words = new ArrayList<String>(); 由于泛型的类型擦除特性,所有添加到words中的变量,都会被转换为Object, 此时就需要泛型的类型转换特性,在执行words.get(i)时做强制转换。
2) 泛型由于类型擦除特性(后面的章节介绍),如List<Integer>,List<String>,List<Number>编译成字节码是一样的,
而不是每一个都有一个版本,保持着简洁性。
3) 最后也是最后要的一点:泛型的兼容性,可兼容jdk1.4以前的版本。从JVM的角度看,由于泛型代码编译成字节码与jdk1.4
之前的代码无任何差别,因此可以很容易的修改jdk1.4写的函数库中的某一个函数,而无需修改整个函数库。
3. 泛型通配符
首先以一个示例讲解为什么需要泛型通配符,如果我们定义addAll方法如下所示:
interface Collection<E> {
...
public boolean addAll(Collection<E> c);
...
}
/**
**测试代码
**/
List<Number> nums = new ArrayList<Number>();
List<Integer> ints = Arrays.asList(1,2,3);
nums.addAll(ints); // 编译错误
如果我们想在List<Number>中添加List<Integer>怎么办呢?这是就需要使用通配符了。
interface Collection<E> {
...
public boolean addAll(Collection<? extends E> c);
...
}
/**
**测试代码
**/
List<Number> nums = new ArrayList<Number>();
List<Integer> ints = Arrays.asList(1,2,3);
List<Double> dbls = Arrays.aslist(3.14,3.21);
nums.addAll(ints);
nums.addAll(dbls);
从上述代码可以看出,通配符的出现解决了泛型的子类问题:
List<Integer>与List<Double>都是List<? extends Number>的子类;而不是 List<Number>的子类。
最后,给出一个结论:当S是T的子类时,List<S>是List<?extends T>的子类;
当S是T的超类时,List<S>是List<? super T>的子类。此外,List<?>是List<? extends Object>的缩写。
接下来,我们分析通配符的重要的特性——put and get原则[2]。
对extends而言,List<? extends T>只能get不能add或put。
List<Integer> ints = new ArrayList<Integer>(); ints.add(1); ints.add(2); List<? extends Number> nums = ints;
System.out.println(nums.get(0)); // 编译成功 nums.add(3); // 编译错误 nums.add(3.14); // 编译错误
对于super而言,List<? super T>只能put或add,不能get。
List<Number> ints = new ArrayList<Number>(); ints.add(1); ints.add(2); List<? super Integer> nums = ints; nums.add(3);
凡是都有例外,对于extends来说,List<? extends Number> nums可以add null,即nums.add(null);对于super来说,
List<? super Integer> nums 可以get(取出)Object对象。
4. 参考文献
[1] http://www.infoq.com/cn/articles/cf-java-generics
[2] java Generics and Collections
java泛型探索——介绍篇的更多相关文章
- Java泛型学习---第二篇
泛型学习第一篇 1.泛型之擦拭法 泛型是一种类似"模板代码"的技术,不同语言的泛型实现方式不一定相同. Java语言的泛型实现方式是擦拭法(Type Erasure). 所谓擦拭法 ...
- java泛型探索——小特性
泛型特性(小篇幅) 1. 补充介绍一些常见的泛型特性: 类型参数T可以是recursive(类似递归性),它的边界可以是类型参数是自身的接口或类. 如我实现寻找最大值的方法,可以这么写: public ...
- Java泛型学习--第一篇
还是那句话,学习某个知识一定要想想为什么要学它,这方面的知识用来解决什么问题的,怎么用,并且要总结的体系化,不能散的到处都是,方便以后查看博客. 今天参考廖雪峰老师官网学习并总结下泛型廖老师官网 1. ...
- java泛型探索——泛型类
本文主要讨论一下如何声明泛型类,讨论的范围涉及构造函数.静态成员.内部类. 构造函数 泛型的类型参数首先声明在首部: public class Pair<T,U> { private fi ...
- Java——泛型
前言 一般的类和方法,使用的都是具体的类型:基本类型或者自定义的类.如果我们要编写出适用于多种类型的通用代码,那么肯定就不能使用具体的类型.前面我们介绍过多态,多态算是一种泛化机制,但是也会拘泥于继承 ...
- 转!!java泛型
介绍java泛型的一篇文章,通俗易懂! 原文地址:http://www.cnblogs.com/lwbqqyumidi/p/3837629.html 一. 泛型概念的提出(为什么需要泛型)? 首先,我 ...
- Java泛型介绍!!!
Java总结篇系列:Java泛型 转自:http://www.cnblogs.com/lwbqqyumidi/p/3837629.html 一. 泛型概念的提出(为什么需要泛型)? 首先,我们看下下 ...
- Java 泛型 介绍
为什么需要泛型? public class GenericTest { public static void main(String[] args) { List list = new ArrayLi ...
- java泛型(一)、泛型的基本介绍和使用
现在开始深入学习java的泛型了,以前一直只是在集合中简单的使用泛型,根本就不明白泛型的原理和作用.泛型在java中,是一个十分重要的特性,所以要好好的研究下. 泛 型的定义:泛型是JDK 1.5的一 ...
随机推荐
- Python可视化学习(1):Matplotlib的配置
Matplotlib是一个优秀的可视化库,它提供了丰富的接口,让Python的可视化落地显得非常容易上手.本系列是本人学习python可视化的学习笔记,主要用于监督自己的学习进度,同时也希望和相关的博 ...
- 大数据时代日志分析平台ELK的搭建
A,首先说说ELK是啥, ELK是ElasticSearch . Logstash 和 Kiabana 三个开源工具组成.Logstash是数据源,ElasticSearch是分析数据的,Kiaba ...
- h5开发app之在线生成二维码
h5通过jquery和qrcode在线生成二维码 首先我们需要下载一个qrcode.js文件,然后依次引入jquery和qrcode文件. 1.创建一个输入框以便做演示使用: <input id ...
- 老李分享:webservice是什么?1
老李分享:webservice是什么? 前言 Web Services 是 Web 应用出于和其他 Web 应用以交互数据为目的的开放式标准(XML.SOAP.HTTP 等).Web Servic ...
- SpringBoot-SpringMvc的Interceptor拦截器配置
Interceptor拦截器实现对每一个用户请求处理前后的业务处理,比如我们需要对用户请求进行响应时间的记录,需要记录请求从开始到结束所耗的时间,这时我们就需要用到拦截器了 下面我们以记录请求处理时间 ...
- C++中的类继承(1) 三种继承方式
继承是使代码可以复用的重要手段,也是面向对象程序设计的核心思想之一.简单的说,继承是指一个对象直接使用另一对象的属性和方法.继承呈现了 面向对象程序设 计的层次结构, 体现了 由简单到复杂的认知过程. ...
- jmeter 使用jmeter 录制web脚本
1.打开jmeter.鼠标右击工作台.添加HTTP代理服务器 2.设置端口号.目标控制器.分组 3.添加查看结果树 4.点击启动.确定完成 5.打开浏览器直接进行操作.就可以看到所录制的脚本信息
- javascript核心概念——new
如果完全没有编程经验的朋友看到这个词会想到什么? 上过幼儿园的都知道new表示 "新的" 的意思. var a = new Date() 按照字面的意思表示什么? 把一个新的dat ...
- 对InvokeRequired的理解
if (listBox1.InvokeRequired) //当有新工作进程访问控件时InvokeRequired为True ...
- Python多层目录模块调用
一. 引用模块在 父+级目录中: 1. 将导入模块所在目录(../model/模块)添加到系统环境变量path下,可添加多个 import syssys.path.append("../mo ...