详解Java泛型之十分钟理解泛型擦除

开发 后端
泛型信息只存在于代码编译阶段,但是在java的运行期(已经生成字节码文件后)与泛型相关的信息会被擦除掉,专业术语叫做类型擦除。

[[408512]]

今天我们来讲解泛型中另一个重要知识点——泛型擦除!

泛型擦除概念

泛型信息只存在于代码编译阶段,但是在java的运行期(已经生成字节码文件后)与泛型相关的信息会被擦除掉,专业术语叫做类型擦除。我们来看一个例子:

  1. ArrayList<Integer> l1 = new ArrayList(); 
  2. ArrayList<String> l2 = new ArrayList();  
  3. System.out.println(l1.getClass()==l2.getClass()); 

 运行代码,结果为True

这是因为ArrayList <String>和ArrayList <Integer>在 jvm 中的 Class 都是 List.class,二者在 jvm 中等同于List<Object> 。

通过下面的例子我们做进一步的分析

  1. import java.lang.reflect.Field; 
  2. public class GeneErasure<T> { 
  3.    T object; 
  4.    public GeneErasure(T object) { 
  5.       this.object = object; 
  6.    } 
  7.    public static void main(String[] args) { 
  8.       GeneErasure demo = new GeneErasure<String>("hi"); 
  9.       Class classz = demo.getClass(); 
  10.       System.out.println(classz.getName()); 
  11.       //输出com.my.generic.GeneErasure 
  12.       Field[] fs = classz.getDeclaredFields(); 
  13.       for ( Field f:fs) { 
  14.         System.out.println("feild: "+f.getName()+"type:"+f.getType().getName()); 
  15.         //输出feild: object type:java.lang.Object 
  16.       } 
  17.    } 

 通过这个例子我们可以看到Class 的类型仍然是GeneErasure并不是GeneErasure <T>这种形式,而类型T被替换成 Object 类型。接下来我们做另一个尝试,把GeneErasure<T> 更改为 GeneErasure < T extends String>

输出结果为:

  1. feild: object type:java.lang.String 

所以,在泛型类被类型擦除的时候,之前泛型类中的类型参数部分如果没有指定上限,如<T>则会被转译成普通的Object 类型,如果指定了上限,如<T extends String>则类型参数就被替换成类型上限。

利用类型擦除干“坏事儿”

大家都知道,下面这段代码l.add(123)无法编译通过,因为123不是String类型,这也是使用泛型的好处之一。

  1. ArrayList<String> l=new ArrayList<String>(); 
  2.  
  3. l.add("abc"); 
  4.  
  5. l.add(123); 

但是我们理解了泛型擦除的原理,我们可以巧妙地利用这个原理结合反射知识干一些“坏事”,例如:

  1. ArrayList<String>l=new ArrayList<String>(); 
  2.      l.add("abc");    
  3.      try { 
  4.        Method method = l.getClass().getDeclaredMethod("add",Object.class); 
  5.        method.invoke(l,"test"); 
  6.        method.invoke(l,100.f); 
  7.      }catch (Exception e) { 
  8.        e.printStackTrace(); 
  9.      } 
  10.      System.out.println("list的大小是:"+l.size());  
  11.      for ( Object o: l){ 
  12.        System.out.println(o); 
  13.      } 

 运行结果是:

  1. list的大小是:3 
  2.  
  3. abc 
  4.  
  5. test 
  6.  
  7. 100.0(被成功插入到ArrayList中) 

我们可以看见100.0 成功地插入到ArrayList <String> 中了,所以利用类型擦除的原理并结合反射的手段就绕过了正常开发中编译器不允许的操作限制。

通俗的理解

我们可以将泛型比作是一个看守,他来守护我们的代码安全,然后设置各项规定,“xxx 禁止出入”的提醒。而现实生活中,也总会有些人能够基于对门卫们生活作息的规律,绕开他们的监视(反射结合泛型擦除)来干一些坏事儿 。

 

责任编辑:姜华 来源: 今日头条
相关推荐

2024-06-19 09:58:29

2013-05-03 10:57:09

泛型泛型教程

2019-04-01 14:59:56

负载均衡服务器网络

2017-11-14 14:41:11

Java泛型IO

2016-06-13 14:07:50

Java动态代理

2020-09-29 06:37:30

Java泛型

2021-06-18 08:25:42

Java泛型通配符

2021-07-09 06:11:37

Java泛型Object类型

2022-03-23 09:32:38

微服务容器Kubernetes

2020-12-17 06:48:21

SQLkafkaMySQL

2021-06-17 06:51:32

Java泛型Java编程

2020-09-27 14:41:37

C语言编程语言计算机

2016-01-04 11:18:00

KubernetesKubernetes概容器技术

2022-06-16 07:31:41

Web组件封装HTML 标签

2021-09-07 09:40:20

Spark大数据引擎

2023-04-12 11:18:51

甘特图前端

2024-05-13 09:28:43

Flink SQL大数据

2015-09-06 09:22:24

框架搭建快速高效app

2012-07-10 01:22:32

PythonPython教程

2023-11-30 10:21:48

虚拟列表虚拟列表工具库
点赞
收藏

51CTO技术栈公众号