摘要:参数化的类型其中是参数化的类型。类型参数的实例或实际类型参数其中是类型参数的实例或实际类型参数。它们并没有重载,而且泛型中也不存在重载这一说法。除此之外,我们应该尽量地多用泛型方法,而减少对整个类的泛化,因为泛型方法更容易把事情说明白。
泛型是适用于许多许多的类型 ---《JAVA编程思想》
在Java的面向对象编程过程中, 或许你知道运用继承、接口等一系列面向对象的动作来实现代码复用,解耦等目的。 但是你知道的,原始类的继承是单继承,而接口更是把类型协议写死了。我们的代码能否不要这么死板(当然你可以认为那是完美的), 我们的代码能否适用于更多的不具体的类型?代码更加通用?甚至这个类型你自己都不知道?,没错,在JDK1.5中,JAVA引入了一个新的代码泛化机制---泛型。
你必须事先知道的一些术语
原始类型
List users=new ArrayList(); 其中List是原始类型。
参数化的类型
Listusers=new ArrayList<>(); 其中List 是参数化的类型。 即把原始类型参数化了。
泛型类型
Listusers=new ArrayList<>(); 其中List 是泛型类型。
类型变量或类型参数
Listusers=new ArrayList<>(); 其中 是类型变量或类型参数。 即一个指向某种类型的变量。 大家知道函数形成参,这里可以看做是"类型形参”。
类型参数的实例或实际类型参数
List泛型也可以如此简单users=new ArrayList<>(); 其中 是类型参数的实例或实际类型参数。 大家知道函数实参,这里可以看做是"类型实参”。
在JDK1.5版本以前,我们使用List集合get数据是需要转型的,如下
List users=new ArrayList<>(); users.add("armstrong"); String user = (String) users.get(0);
上面代码中,List集合是可以传入任何Object及它的子类的。我们传入一个String类型的数据,取出来默认是Object,所以你需要转型。使用原始类型的弊端:在程序运行起来的时候,可能会出现转型失败的异常。那么补救方法呢?
Jdk1.5里面是这样定义List:
public interface Listextends Collection { int size(); boolean isEmpty(); T[] toArray(T[] a); boolean add(E e); ... }
你可以看到List类这个原始类型,定义成了接口List泛型类型。类、接口的泛型定义方式是一样的,那么我们可以这样编写之前那段代码:
ListusersGenertor=new ArrayList<>(); usersGenertor.add("armstrong"); String userS = usersGenertor.get(0);
而且,此时List
ListusersGenertor=new ArrayList<>(); usersGenertor.add(new Object());// error,此处不能添加Object类型 usersGenertor.add(123);// error,此处不能添加Integer类型 String userS = usersGenertor.get(0);
为了更好的解释泛型的概念。打个比方,我们知道达尔文的《进化论》中说人和猴子的祖先都是猿。但是人只能生人 猴子只能生猴子,虽然有共同的祖先,但是人类不等于猴类 我们也不能说人是猿。人不等于猿~ 那么List后面跟着一个类型变量T,就是指List只能持有指定类型的元素,如果T如果被String替代,那么List就只能持有String, 而List
List
有同学会说,这个例子太简单了,能不能来一些有难度的。 其实,我举这个例子的初衷是让大家知道---JDK开发人员引入泛型直到现在最大的好处就是为了解决容器转型的麻烦,而且平时工作中我们用泛型最多的场景是和容器打交道。
非常好用的泛型方法在定义List的接口源码中我们可以看到如下方法:
T[] toArray(T[] a);
这个方法的作用是把List中的所有元素转存到另一个指定类型的数组中。List原本装什么类型的元素,那么数组也装同样类型的元素,并且这里传入的类型和返回值类型一致都是类型变量T[]。
那么泛型方法是如何定义的呢? 在上面代码中,我们在返回值的前面,用
ListusersGenertor=new ArrayList<>(); usersGenertor.add("zeng"); usersGenertor.add("armstrong"); String[] strings = usersGenertor.toArray(new String[10]); for (String string : strings) { System.out.println("string:_"+string); } List usersGenertor2=new ArrayList<>(); usersGenertor2.add(1); usersGenertor2.add(2); Integer[] ints = usersGenertor2.toArray(new Integer[3] ); for (Integer inta : ints) { System.out.println("Integer:_"+inta); } output: string:_zeng string:_armstrong string:_null string:_null string:_null string:_null string:_null string:_null string:_null string:_null Integer:_1 Integer:_2 Integer:_null
上面的代码中toArray方法分别传入了String和Integer两种不同的引用类型。它们并没有重载,而且泛型中也不存在重载这一说法。暂时我们可以简单的想象成String,Integer在编译期间把T这个类型参数变量替换掉了。
除此之外,我们应该尽量地多用泛型方法,而减少对整个类的泛化,因为泛型方法更容易把事情说明白。
参考资料:
《Java编程思想第四版》
《Effective Java》
文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。
转载请注明本文地址:https://www.ucloud.cn/yun/65641.html
摘要:静态变量是被泛型类的所有实例所共享的。所以引用能完成泛型类型的检查。对于这个类型系统,有如下的一些规则相同类型参数的泛型类的关系取决于泛型类自身的继承体系结构。事实上,泛型类扩展都不合法。 前言 和C++以模板来实现静多态不同,Java基于运行时支持选择了泛型,两者的实现原理大相庭径。C++可以支持基本类型作为模板参数,Java却只能接受类作为泛型参数;Java可以在泛型类的方法中取得...
摘要:可以看到,如果我们给泛型类制定了上限,泛型擦除之后就会被替换成类型的上限。相应的,泛型类中定义的方法的类型也是如此。参考语言类型擦除下界通配符和的区别 本篇博客主要介绍了Java类型擦除的定义,详细的介绍了类型擦除在Java中所出现的场景。 1. 什么是类型擦除 为了让你们快速的对类型擦除有一个印象,首先举一个很简单也很经典的例子。 // 指定泛型为String List list1 ...
摘要:泛型类在类的申明时指定参数,即构成了泛型类。换句话说,泛型类可以看成普通类的工厂。的作用就是指明泛型的具体类型,而类型的变量,可以用来创建泛型类的对象。只有声明了的方法才是泛型方法,泛型类中的使用了泛型的成员方法并不是泛型方法。 什么是泛型? 泛型是JDK 1.5的一项新特性,它的本质是参数化类型(Parameterized Type)的应用,也就是说所操作的数据类型被指定为一个参数,...
阅读 2303·2021-09-28 09:45
阅读 3594·2021-09-24 09:48
阅读 2255·2021-09-22 15:49
阅读 3092·2021-09-08 16:10
阅读 1585·2019-08-30 15:54
阅读 2316·2019-08-30 15:53
阅读 3012·2019-08-29 18:42
阅读 2863·2019-08-29 16:19