Java 中一个天天都在被人使用,但你并不知道为什么的知识点

开发 架构
泛型,就是泛化类型也就是泛化参数类型。平时我们在编写代码的时候,方法的参数在定义的时候都是指定特定的类型,比如 Integer,Double 或者其他自己编写的类。

[[432374]]

泛型作为 Java 中一个天天都在被人使用的特性,你真的知道它的原理吗?

什么是泛型

首先我们说下什么是泛型。

泛型,就是泛化类型也就是泛化参数类型。平时我们在编写代码的时候,方法的参数在定义的时候都是指定特定的类型,比如 Integer,Double 或者其他自己编写的类。那么泛化类型就是,我们在编写一个方法的时候对于参数的类型不具体指定,而是定义一个通用类型,在使用的时候根据类型自动转化。

上面的描述可能比较抽象,我们再看一下,如果没有泛型的话,会出现什么情况以及为什么说这个泛型大家天天都在使用。

原理

我们都知道 ArrayList 作为 Java 中一个很频繁被使用的集合,它是一个可变长的数组,底层是基于 Object[] 来实现的。

可以简单理解为下面的内容

  1. public class ArrayList { 
  2.     private Object[] array; 
  3.     private int size
  4.     public void add(Object e) {...} 
  5.     public void remove(int index) {...} 
  6.     public Object get(int index) {...} 

如果说这个时候我们使用上面的 ArrayList 去存储 String 类型的话,需要如下操作,在使用的时候必须进行手动强转。

  1. ArrayList list = new ArrayList(); 
  2. list.add("Java"); 
  3. list.add("C++"); 
  4. String first = (String) list.get(0); 
  5. String first = (String) list.get(1); 

首先看到上面的代码,大家一定会诧异,要是每次使用的时候都这样显示强转的话,那不是要命了么,而且这还是使用者知道是什么类型的情况才能进行手动强转,如果说根本不知道是什么类型的时候,根据没办法进行强转,这种方式简直不能忍,还特别容易出错。

那怎么解决这个问题呢?有朋友说我们可以对于不同的类型实现一个自己的 ArrayList 类,这样在使用的时候就可以不用强转了啊。对此阿粉只能说,对于 JDK 提供的类可以这样做,但是对于用户自己编写的类怎么实现呢?

这个时候大家可能会说到,ArrayList 我天天使用,也没手动强转过啊,不还是用的好好的。

这就要归功于我们今天所说的主角,泛型了。

我们给 ArrayList 增加的泛型,通过定义一个泛化的类型,当我们在使用的时候如果传递的类型不是指定的类型,那么在编译的阶段就会报错,从而也就不会有需要强转的操作了。

  1. public class ArrayList<E> { 
  2.     private Object[] array;//任何类型都是 Object 的子类,所以这里我们还是不变 
  3.     private int size
  4.     public void add(E e) {...} 
  5.     public void remove(int index) {...} 
  6.     public E get(int index) {...} 

这样修改过后,我们在编写代码的时候就可以如果进行

  1. ArrayList<String> strList = new ArrayList<String>(); 
  2. list.add("Java"); 
  3. list.add("C++"); 
  4. String first = list.get(0);//这里就不用强转了 
  5. String first = list.get(1);//这里就不用强转了 
  6. list.add(new Integer(100));//编译报错 

当我们需要使用 Integer 对象的时候就可以使用下面这种方式

  1. ArrayList<Integer> list = new ArrayList<Integer>(); 
  2. list.add("Java");//编译报错 
  3. list.add("C++");//编译报错 
  4. list.add(new Integer(100));//编译通过 

另外我们还知道 ArrayList 实现了 List 接口,如下所示,所以会有一种向上转型的概念,就是我们前面在定义的时候使用 List 也是可以,也就是我们通常的定义方式,即 List list = new ArrayList<>();

但是这里我们需要注意不可以进行如下的泛型向上转型,比如下面这个例子。

我们定义了 Person 类,Man 类以及 Women 类

  1. public class Person { 
  2.  
  3.     private String name
  4.     private Integer age; 
  5.  
  6.     public String getName() { 
  7.         return name
  8.     } 
  9.  
  10.     public void setName(String name) { 
  11.         this.name = name
  12.     } 
  13.  
  14.     public Integer getAge() { 
  15.         return age; 
  16.     } 
  17.  
  18.     public void setAge(Integer age) { 
  19.         this.age = age; 
  20.     } 
  21.  
  22. public class Man extends Person { 
  23.  
  24.     ... 
  25.  
  26. public class Women extends Person { 
  27.  
  28.     ... 

我们在使用的时候只能这样

  1. ArrayList<Man> manList = new ArrayList<Man>(); 
  2. List<Man> manList1 = new ArrayList<>(); 
  3. ArrayList<Women> womenList = new ArrayList<Women>(); 
  4. List<Women> womenList1 = new ArrayList<>(); 

不可以

  1. ArrayList<Man> manList = new ArrayList<Man>(); 
  2. //这种转型是不可以的 
  3. ArrayList<Person> personList = manList; 
  4. personList.add(new Man()); 
  5. //破坏了原本只能存放 Man 的约定 
  6. personList.add(new Women()); 

因为我们不能同时在一个List 中即加入 man 也加入 woman,这样是不行的。

接下来我们再看另一个问题,假设我们有一个方法,是打印 PersonList 内容的,如下所示:

  1. public void print(ArrayList<Person> personList) { 
  2.     for (Person p : personList) { 
  3.         System.out.print(p.name); 
  4.     } 
  5. ArrayList<Man> manList = new ArrayList(); 
  6. list.add(new Man()); 
  7. list.add(new Man()); 
  8. print(manList); 

上面的内容会编译出错,效果是这样的。

原因是因为虽然 Man 类是继承了 Person 类,但是 ArrayList 并没有继承ArrayList 类,所以这个方法是编译通不过。看到这里小伙伴又惊呆了,这不行啊,总不能引入了泛型,就不支持多态了吧,所以这个要怎么办。

这里我们就需要引入另一个东西了,那就是泛型里面的 extends,我们把 print 方法换个写法,这个时候就不会编译不通过了。如下所示图片

 

extends 表示传进来的参数只要是 Person 的子类都可以,这样就还支持多态了。所以现在小伙伴知道了为什么JDK 源码以及很多框架的源码中都有很多? extends xxx 这种形式的代码了吧。

 

责任编辑:武晓燕 来源: Java极客技术
相关推荐

2024-06-04 08:09:00

kubernetesHPA扩缩容

2016-09-13 17:14:30

2022-10-27 21:01:03

JavaScrip面试题开发

2011-08-16 13:15:15

MongoDB

2021-11-16 07:31:59

JavaJDK 7 String

2019-08-16 16:11:01

消息队列MQ解耦

2019-08-09 14:20:46

微信软件手机

2018-04-26 13:10:06

人工智能机器人京东

2022-10-12 08:22:44

Guava工具Collection

2021-02-24 09:45:21

数据库工具技术

2015-05-14 15:59:33

DockerLinux容器管理工具

2018-10-28 15:40:23

Python编程语言

2011-04-01 09:49:54

Python

2022-05-16 07:31:51

Java进度条代码

2010-08-23 09:56:09

Java性能监控

2018-10-17 09:25:22

2014-01-22 16:19:06

游戏圈游戏创业移动游戏

2024-04-03 09:23:31

ES索引分析器

2019-10-11 10:03:02

负载均衡器LVS虚拟服务器

2020-08-11 11:20:49

Linux命令使用技巧
点赞
收藏

51CTO技术栈公众号