一. 泛型概念的提出(为什么需要泛型)?

首先,我们看下下面这段简短的代码:

1 public classGenericTest {2

3 public static voidmain(String[] args) {4 List list = newArrayList();5 list.add("qqyumidi");6 list.add("corn");7 list.add(100);8

9 for (int i = 0; i < list.size(); i++) {10 String name = (String) list.get(i); //1

11 System.out.println("name:" +name);12 }13 }14 }

定义了一个List类型的集合,先向其中加入了两个字符串类型的值,随后加入一个Integer类型的值。这是完全允许的,因为此时list默认的类型为Object类型。在之后的循环中,由于忘记了之前在list中也加入了Integer类型的值或其他编码原因,很容易出现类似于//1中的错误。因为编译阶段正常,而运行时会出现“java.lang.ClassCastException”异常。因此,导致此类错误编码过程中不易发现。

在如上的编码过程中,我们发现主要存在两个问题:

1.当我们将一个对象放入集合中,集合不会记住此对象的类型,当再次从集合中取出此对象时,改对象的编译类型变成了Object类型,但其运行时类型任然为其本身类型。

2.因此,//1处取出集合元素时需要人为的强制类型转化到具体的目标类型,且很容易出现“java.lang.ClassCastException”异常。

那么有没有什么办法可以使集合能够记住集合内元素各类型,且能够达到只要编译时不出现问题,运行时就不会出现“java.lang.ClassCastException”异常呢?答案就是使用泛型。

二.什么是泛型?

泛型,即“参数化类型”。一提到参数,最熟悉的就是定义方法时有形参,然后调用此方法时传递实参。那么参数化类型怎么理解呢?顾名思义,就是将类型由原来的具体的类型参数化,类似于方法中的变量参数,此时类型也定义成参数形式(可以称之为类型形参),然后在使用/调用时传入具体的类型(类型实参)。

看着好像有点复杂,首先我们看下上面那个例子采用泛型的写法。

1 public classGenericTest {2

3 public static voidmain(String[] args) {4 /*

5 List list = new ArrayList();6 list.add("qqyumidi");7 list.add("corn");8 list.add(100);9 */

10

11 List list = new ArrayList();12 list.add("qqyumidi");13 list.add("corn");14 //list.add(100);//1 提示编译错误

15

16 for (int i = 0; i < list.size(); i++) {17 String name = list.get(i); //2

18 System.out.println("name:" +name);19 }20 }21 }

采用泛型写法后,在//1处想加入一个Integer类型的对象时会出现编译错误,通过List,直接限定了list集合中只能含有String类型的元素,从而在//2处无须进行强制类型转换,因为此时,集合能够记住元素的类型信息,编译器已经能够确认它是String类型了。

结合上面的泛型定义,我们知道在List中,String是类型实参,也就是说,相应的List接口中肯定含有类型形参。且get()方法的返回结果也直接是此形参类型(也就是对应的传入的类型实参)。下面就来看看List接口的的具体定义:

1 public interface List extends Collection{2

3 intsize();4

5 booleanisEmpty();6

7 booleancontains(Object o);8

9 Iteratoriterator();10

11 Object[] toArray();12

13 T[] toArray(T[] a);14

15 booleanadd(E e);16

17 booleanremove(Object o);18

19 boolean containsAll(Collection>c);20

21 boolean addAll(Collection extends E>c);22

23 boolean addAll(int index, Collection extends E>c);24

25 boolean removeAll(Collection>c);26

27 boolean retainAll(Collection>c);28

29 voidclear();30

31 booleanequals(Object o);32

33 inthashCode();34

35 E get(intindex);36

37 E set(intindex, E element);38

39 void add(intindex, E element);40

41 E remove(intindex);42

43 intindexOf(Object o);44

45 intlastIndexOf(Object o);46

47 ListIteratorlistIterator();48

49 ListIterator listIterator(intindex);50

51 List subList(int fromIndex, inttoIndex);52 }

我们可以看到,在List接口中采用泛型化定义之后,中的E表示类型形参,可以接收具体的类型实参,并且此接口定义中,凡是出现E的地方均表示相同的接受自外部的类型实参。

自然的,ArrayList作为List接口的实现类,其定义形式是:

1 public class ArrayList extends AbstractList

2 implements List, RandomAccess, Cloneable, java.io.Serializable {3

4 public booleanadd(E e) {5 ensureCapacityInternal(size + 1); //Increments modCount!!

6 elementData[size++] =e;7 return true;8 }9

10 public E get(intindex) {11 rangeCheck(index);12 checkForComodification();13 return ArrayList.this.elementData(offset +index);14 }15

16 //...省略掉其他具体的定义过程

17

18 }

由此,我们从源代码角度明白了为什么//1处加入Integer类型对象编译错误,且//2处get()到的类型直接就是String类型了。

三.自定义泛型接口、泛型类和泛型方法

从上面的内容中,大家已经明白了泛型的具体运作过程。也知道了接口、类和方法也都可以使用泛型去定义,以及相应的使用。是的,在具体使用时,可以分为泛型接口、泛型类和泛型方法。

自定义泛型接口、泛型类和泛型方法与上述Java源码中的List、ArrayList类似。如下,我们看一个最简单的泛型类和方法定义:

1 public classGenericTest {2

3 public static voidmain(String[] args) {4

5 Box name = new Box("corn");6 System.out.println("name:" +name.getData());7 }8

9 }10

11 class Box{12

13 privateT data;14

15 publicBox() {16

17 }18

19 publicBox(T data) {20 this.data =data;21 }22

23 publicT getData() {24 returndata;25 }26

27 }

在泛型接口、泛型类和泛型方法的定义过程中,我们常见的如T、E、K、V等形式的参数常用于表示泛型形参,由于接收来自外部使用时候传入的类型实参。那么对于不同传入的类型实参,生成的相应对象实例的类型是不是一样的呢?

1 public classGenericTest {2

3 public static voidmain(String[] args) {4

5 Box name = new Box("corn");6 Box age = new Box(712);7

8 System.out.println("name class:" + name.getClass()); //com.qqyumidi.Box

9 System.out.println("age class:" + age.getClass()); //com.qqyumidi.Box

10 System.out.println(name.getClass() == age.getClass()); //true

11

12 }13

14 }

由此,我们发现,在使用泛型类时,虽然传入了不同的泛型实参,但并没有真正意义上生成不同的类型,传入不同泛型实参的泛型类在内存上只有一个,即还是原来的最基本的类型(本实例中为Box),当然,在逻辑上我们可以理解成多个不同的泛型类型。

究其原因,在于Java中的泛型这一概念提出的目的,导致其只是作用于代码编译阶段,在编译过程中,对于正确检验泛型结果后,会将泛型的相关信息擦出,也就是说,成功编译过后的class文件中是不包含任何泛型信息的。泛型信息不会进入到运行时阶段。

对此总结成一句话:泛型类型在逻辑上看以看成是多个不同的类型,实际上都是相同的基本类型。

四.类型通配符

接着上面的结论,我们知道,Box和Box实际上都是Box类型,现在需要继续探讨一个问题,那么在逻辑上,类似于Box和Box是否可以看成具有父子关系的泛型类型呢?

为了弄清这个问题,我们继续看下下面这个例子:

1 public classGenericTest {2

3 public static voidmain(String[] args) {4

5 Box name = new Box(99);6 Box age = new Box(712);7

8 getData(name);9

10 //The method getData(Box) in the type GenericTest is11 //not applicable for the arguments (Box)

12 getData(age); // 113

14 }15

16 public static void getData(Boxdata){17 System.out.println("data :" +data.getData());18 }19

20 }

我们发现,在代码//1处出现了错误提示信息:The method getData(Box) in the t ype GenericTest is not applicable for the arguments (Box)。显然,通过提示信息,我们知道Box在逻辑上不能视为Box的父类。那么,原因何在呢?

1 public classGenericTest {2

3 public static voidmain(String[] args) {4

5 Box a = new Box(712);6 Box b = a; //1

7 Box f = new Box(3.14f);8 b.setData(f); //2

9

10 }11

12 public static void getData(Boxdata) {13 System.out.println("data :" +data.getData());14 }15

16 }17

18 class Box{19

20 privateT data;21

22 publicBox() {23

24 }25

26 publicBox(T data) {27 setData(data);28 }29

30 publicT getData() {31 returndata;32 }33

34 public voidsetData(T data) {35 this.data =data;36 }37

38 }

这个例子中,显然//1和//2处肯定会出现错误提示的。在此我们可以使用反证法来进行说明。

假设Box在逻辑上可以视为Box的父类,那么//1和//2处将不会有错误提示了,那么问题就出来了,通过getData()方法取出数据时到底是什么类型呢?Integer? Float? 还是Number?且由于在编程过程中的顺序不可控性,导致在必要的时候必须要进行类型判断,且进行强制类型转换。显然,这与泛型的理念矛盾,因此,在逻辑上Box不能视为Box的父类。

好,那我们回过头来继续看“类型通配符”中的第一个例子,我们知道其具体的错误提示的深层次原因了。那么如何解决呢?总部能再定义一个新的函数吧。这和Java中的多态理念显然是违背的,因此,我们需要一个在逻辑上可以用来表示同时是Box和Box的父类的一个引用类型,由此,类型通配符应运而生。

类型通配符一般是使用 ? 代替具体的类型实参。注意了,此处是类型实参,而不是类型形参!且Box>在逻辑上是Box、Box...等所有Box的父类。由此,我们依然可以定义泛型方法,来完成此类需求。

1 public classGenericTest {2

3 public static voidmain(String[] args) {4

5 Box name = new Box("corn");6 Box age = new Box(712);7 Box number = new Box(314);8

9 getData(name);10 getData(age);11 getData(number);12 }13

14 public static void getData(Box>data) {15 System.out.println("data :" +data.getData());16 }17

18 }

有时候,我们还可能听到类型通配符上限和类型通配符下限。具体有是怎么样的呢?

在上面的例子中,如果需要定义一个功能类似于getData()的方法,但对类型实参又有进一步的限制:只能是Number类及其子类。此时,需要用到类型通配符上限。

1 public classGenericTest {2

3 public static voidmain(String[] args) {4

5 Box name = new Box("corn");6 Box age = new Box(712);7 Box number = new Box(314);8

9 getData(name);10 getData(age);11 getData(number);12

13 //getUpperNumberData(name);//1

14 getUpperNumberData(age); //2

15 getUpperNumberData(number); //3

16 }17

18 public static void getData(Box>data) {19 System.out.println("data :" +data.getData());20 }21

22 public static void getUpperNumberData(Box extends Number>data){23 System.out.println("data :" +data.getData());24 }25

26 }

此时,显然,在代码//1处调用将出现错误提示,而//2 //3处调用正常。

类型通配符上限通过形如Box extends Number>形式定义,相对应的,类型通配符下限为Box super

Number>形式,其含义与类型通配符上限正好相反,在此不作过多阐述了。

五.话外篇

本文中的例子主要是为了阐述泛型中的一些思想而简单举出的,并不一定有着实际的可用性。另外,一提到泛型,相信大家用到最多的就是在集合中,其实,在实际的编程过程中,自己可以使用泛型去简化开发,且能很好的保证代码质量。并且还要注意的一点是,Java中没有所谓的泛型数组一说。

对于泛型,最主要的还是需要理解其背后的思想和目的。

java实现lbs_Java总结篇系列:Java泛型相关推荐

  1. Java私塾跟我学系列——JAVA篇 第四章Java类和对象

    教学目标: i面向对象基础 i掌握对象的三大特性 i掌握Java类的构建 i掌握如何使用Java类 i理解引用类型 i理解按值传递和按引用传递 i深入理解变量 i掌握包装类 i理解类型转换 i理解Ja ...

  2. Java基础与提高干货系列——Java反射机制

    前言 今天介绍下Java的反射机制,以前我们获取一个类的实例都是使用new一个实例出来.那样太low了,今天跟我一起来学习学习一种更加高大上的方式来实现. 正文 Java反射机制定义 Java反射机制 ...

  3. java 坦克大战画坦克_【JAVA语言程序设计基础篇】--JAVA实现坦克大战游戏--画出坦克(二)...

    坦克类: //坦克父类 可以设置坦克出现位置(x,y) class Tank { int x = 0; int y = 0; public Tank (int x,int y){ this.x = x ...

  4. java私塾 java篇_Java私塾跟我学系列——JAVA篇 五、

    五:Java如何做到让机器理解我们想要做的东西 用一个图来描述这个过程会比较容易理解: 1:编写代码 首先把我们想要计算机做的事情,通过Java表达出来,写成Java文件,这个过程就是 编写代码的过程 ...

  5. Java新手小白入门篇 Java基础(一)

    Java新手小白入门篇 Java基础 Java新手小白入门篇 Java基础(知识点体系汇总) Java新手小白入门篇 Java基础(一) Java新手小白入门篇 Java基础(二) Java新手小白入 ...

  6. Java 9的误解和Java 10的愿望清单:Java影响者的全部访谈

    是时候消除那些Java 9的误解了 Java 9将在9月到达,即使我们已经准备好了也没有,但是在到达Java 9之前,仍有一些事情需要理解. 我们需要停止思考Maven在Java 9上不起作用,并且如 ...

  7. Java总结篇系列:Java多线程(三)

    2019独角兽企业重金招聘Python工程师标准>>> 本文主要接着前面多线程的两篇文章总结Java多线程中的线程安全问题. 一.一个典型的Java线程安全例子 public cla ...

  8. Java总结篇系列:Java多线程(二)

    本文承接上一篇文章<Java总结篇系列:Java多线程(一)>. 四.Java多线程的阻塞状态与线程控制 上文已经提到Java阻塞的几种具体类型.下面分别看下引起Java线程阻塞的主要方法 ...

  9. java = 优化_Java9系列第7篇:Java.util.Optional优化与增强 - 字母哥博客 - 博客园

    我计划在后续的一段时间内,写一系列关于java 9的文章,虽然java 9 不像Java 8或者Java 11那样的核心java版本,但是还是有很多的特性值得关注.期待您能关注我,我将把java 9 ...

最新文章

  1. iOS 项目集成Flutter
  2. 小小的吹一下集结号~
  3. SWT 下菜单与子菜单添加的实现(详细图解)
  4. mybatis 动态 SQL 官方文档
  5. 2022年3月14日蓝桥杯基础算法能力测试
  6. beego 例子_beego框架代码分析
  7. java中的过滤器与监听器
  8. APP开发内容介绍(源代码)
  9. c语言中有队列头文件吗,C语言队列学习竟是如此简单!你,懂了嘛?
  10. cpri带宽不足的解决方法_CPRI基带数据的低损耗压缩
  11. python爬虫--代理的使用
  12. 计算机如何更改扩展名,文件扩展名怎么改,怎样更改文件的扩展名
  13. Zotero 使用技巧
  14. 视频和视频帧:图像,从自然光到01串
  15. Java实现 蓝桥杯VIP 算法提高 分苹果
  16. 圣诞邮件怎么写?收藏住~
  17. 职业自我认知的测试软件,职业生涯规划自我认知测试.docx
  18. 推荐10本EEG领域值得阅读的书籍
  19. Androidstudio报错问题之R飘红
  20. 二维码生成:STM32F407 + LCD屏(GUI)制作二维码

热门文章

  1. Android 车机初体验:Auto,Automotive 傻傻分不清楚?
  2. 时间加减计算器_初级会计职称考试不让带计算器?!手把手教你使用机考系统计算器,再不看就晚了!...
  3. 碧蓝航线内部表情包(有爱自取)
  4. vue使用echarts-liquidfill水球图不生效
  5. 一键配置网站服务器环境,服务器环境一键配置
  6. 微信小程序云开发项目——多肉植物销售小程序
  7. element实现el-progress线形进度条渐变色
  8. C/C++ 学习日记3:制作带有界面的植物大战僵尸助手(基于MFC)
  9. sublime_字体更换
  10. win10系统AMD显卡OBS录屏黑屏解决方法