美文网首页测试开发Java编码知识点测试开发那些事儿
详解Java泛型之3——十分钟理解泛型擦除

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

作者: 测试开发Kevin | 来源:发表于2021-06-18 10:16 被阅读0次


前面我们介绍了泛型以及通配符的基础概念,可以参考文章:

详解Java泛型之1——入门泛型必懂的知识点

详解Java泛型之2——详解通配符

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

泛型擦除概念

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

ArrayList<Integer> l1 = new ArrayList();

ArrayList<String> l2 = new ArrayList(); 

System.out.println(l1.getClass()==l2.getClass());

运行代码,结果为True

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

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

import java.lang.reflect.Field;

public class GeneErasure<T> {

   T object;

   public GeneErasure(T object) {

      this.object = object;

   }

   public static void main(String[] args) {

      GeneErasure demo = new GeneErasure<String>("hi");

      Class classz = demo.getClass();

      System.out.println(classz.getName());

      //输出com.my.generic.GeneErasure

      Field[] fs = classz.getDeclaredFields();

      for ( Field f:fs) {

        System.out.println("feild: "+f.getName()+"type:"+f.getType().getName());

        //输出feild: object type:java.lang.Object

      }

   }

通过这个例子我们可以看到Class 的类型仍然是GeneErasure并不是GeneErasure <T>这种形式,而类型T被替换成 Object 类型。

接下来我们做另一个尝试,把GeneErasure <T> 更改为 GeneErasure < T extends String>

输出结果为feild: object type:java.lang.String

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

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

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

ArrayList<String> l=new ArrayList<String>();

l.add("abc");

l.add(123);

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

 ArrayList<String>l=new ArrayList<String>();

      l.add("abc");   

      try {

        Method method = l.getClass().getDeclaredMethod("add",Object.class);

        method.invoke(l,"test");

        method.invoke(l,100.f);

      }catch (Exception e) {

        e.printStackTrace();

      }

      System.out.println("list的大小是:"+l.size()); 

      for ( Object o: l){

        System.out.println(o);

      }

运行结果是:

list的大小是:3

abc

test

100.0(被成功插入到ArrayList中)

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

通俗的理解

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

相关文章

  • 详解Java泛型之4——一个例子理解泛型带来的好处

    前面我介绍了关于泛型、通配符以及泛型擦除的相关知识点,大家可以参考以下文章: 详解Java泛型之1——入门泛型必懂...

  • 【进阶之路】Java的类型擦除式泛型

    【进阶之路】Java的类型擦除式泛型 Java选择的泛型类型叫做类型擦除式泛型。什么是类型擦除式泛型呢?就是Jav...

  • Android 开发也要掌握的Java知识 - Java泛型

    如果需要看泛型擦除Java泛型擦除 1.Java泛型有什么用?为啥要使用泛型? Java中数组的类型是定义的时候就...

  • java_泛型

    类型擦除正确理解泛型概念的首要前提是理解类型擦除(type erasure)。 Java中的泛型基本上都是在编译器...

  • Java 泛型擦除原理

    问:请比较深入的谈谈你对 Java 泛型擦除的理解和带来的问题认识? 答:Java 的泛型是伪泛型,因为在编译期间...

  • java泛型

    java的泛型是"伪泛型",为什么这么说。因为泛型只是作用在编译之前,编译之后,泛型都被擦除了(类型擦除)。所以说...

  • Kotlin语言(六):泛型

    1、泛型类 2、泛型函数 3、泛型上限 4、泛型擦除 5、泛型投射 6、星号投射

  • Java泛型:类型擦除

    前情回顾 Java泛型:泛型类、泛型接口和泛型方法 类型擦除 代码片段一 显然在平时使用中,ArrayList (...

  • Java如何在运行时获取泛型的类型

    Java泛型是伪泛型,会在编译完成时进行类型的擦除,我们无法在运行时获取泛型参数的具体类型(类型擦除会被替换成泛型...

  • 泛型

    java 泛型详解-绝对是对泛型方法讲解最详细的,没有之一Java泛型深入理解加泛型面试数组的协变性与范型的不可变性

网友评论

    本文标题:详解Java泛型之3——十分钟理解泛型擦除

    本文链接:https://www.haomeiwen.com/subject/ankkyltx.html