详解Java 泛型

一、什么是泛型

Java 泛型(generics)是 JDK 5 中引入的一个新特性, 泛型提供了编译时类型安全检测机制,该机制允许程序员在编译时检测到非法的类型。

简单理解就是:泛型指定编译时的类型,减少运行时由于对象类型不匹配引发的异常。其主要用途是提高我们的代码的复用率。

我们Java标准库中的ArrayList就是泛型使用的典型应用:

public class ArrayList extends AbstractList implements List, RandomAccess, Cloneable, java.io.Serializable {
       
     ......

   public ArrayList(Collection c) {
       elementData = c.toArray();
       if ((size = elementData.length) != 0) {
           // c.toArray might (incorrectly) not return Object[] (see 6260652)
           if (elementData.getClass() != Object[].class)
               elementData = Arrays.copyOf(elementData, size, Object[].class);
       } else {
           // replace with empty array.
           this.elementData = EMPTY_ELEMENTDATA;
       }
   }

   public void sort(Comparator c) {
       final int expectedModCount = modCount;
       Arrays.sort((E[]) elementData, 0, size, c);
       if (modCount != expectedModCount) {
           throw new ConcurrentModificationException();
       }
       modCount++;
   }
   
 .....

   public E get(int index) {
       rangeCheck(index);

       return elementData(index);
   }

   public boolean add(E e) {
       ensureCapacityInternal(size + 1);  // Increments modCount!!
       elementData[size++] = e;
       return true;
   }

}

源码中,ArrayList中的E称为类型参数变量,而整个ArrayList我们称为泛型类型。 我们可以指定除基本类型之外的任何类型,如:ArrayList。

源码中Collection 中? 通配符类型 表示类型的上界,表示参数化类型的可能是T 或是 T的子类。

源码中Comparator 表示类型下界(Java Core中叫超类型限定),表示参数化类型是此类型的超类型(父类型),直至Object。

二、extends和super通配符

在定义泛型类型Generic的时候,也可以使用extends通配符来限定T的类型:

public class Generic { ... }

现在,我们只能定义:

Generic p1 = null;
Generic p2 = new Generic(1, 2);
Generic p3 = null;

因为Number、Integer和Double都符合。

非Number类型将无法通过编译:

Generic p1 = null; // compile error!
Generic

因为String、Object都不符合,因为它们不是Number类型或Number的子类。

我们看一个例子:

public class Test {

   static class Food {

   }

   static class Fruit extends Food {
   }

   static class Apple extends Fruit {
   }

   static class Orange extends Fruit {
   }

   public void testExtend() {
       List list = new ArrayList();

       //无法安全添加任何具有实际意义的元素,报错,extends为上界通配符,只能取值,不能放.
       //因为Fruit的子类不只有Apple还有Orange,这里不能确定具体的泛型到底是Apple还是Orange,所以放入任何一种类型都会报错

       //list.add(new Apple());
       //list.add(new Orange());

       //可以添加null,因为null可以表示任何类型
       list.add(null);

       //可以正常获取,用java多态
       Food foot = list.get(0);
       Apple apple = (Apple) list.get(0);
   }

   public void testSuper() {
       List list = new ArrayList();

       //super为下界通配符,可以存放元素,但是也只能存放当前类或者子类的实例,以当前的例子来讲,
       list.add(new Fruit());
       list.add(new Apple());

       //无法确定Fruit的父类是否只有Food一个(Object是超级父类)
       //因此放入Food的实例编译不通过,只能放自己的实例 或者根据java多态的特性放子类实例
       //list.add(new Food());
       //List list2 = new ArrayList();
       //Fruit fruit = list.get(0); //不能确定返回类型

   }

}

在testExtend方法中,因为泛型中用的是extends,在向list中存放元素的时候,我们并不能确定List中的元素的具体类型,即可能是Apple也可能是Orange。因此调用add方法时,不论传入new Apple()还是new Orange(),都会出现编译错误。

理解了extends之后,再看super就很容易理解了,即我们不能确定testSuper方法的参数中的泛型是Fruit的哪个父类,因此在调用get方法时只能返回Object类型。结合extends可见,在获取泛型元素时,使用extends获取到的是泛型中的上边界的类型(本例子中为Fruit),范围更小。

总结:

在使用泛型时,存取元素时用super。

获取元素时,用extends。

有了上面的结论我们看下Java标准库的Collections类定义的copy()方法,这个copy()方法的定义就完美地展示了extends和super的意图:

copy()方法内部不会读取dest,因为不能调用dest.get()来获取T的引用;

copy()方法内部也不会修改src,因为不能调用src.add(T)。

public class Collections {
   // 把src的每个元素复制到dest中:
   public static  void copy(List dest, List src) {
       for (int i=0; i

三、泛型擦除

Java的泛型是伪泛型,这是因为Java在编译期间,所有的泛型信息都会被擦掉,正确理解泛型概念的首要前提是理解类型擦除。Java的泛型基本上都是在编译器这个层次上实现的,在生成的字节码中是不包含泛型中的类型信息的,使用泛型的时候加上类型参数,在编译器编译的时候会去掉,这个过程成为类型擦除

我们看一个示例:

public class Test2 {

   public static void main(String[] args) {
       Map map = new HashMap();
       Animal animal = new Animal();
       animal.setVegetarian(true);
       animal.setEats("fish");
       map.put("cat", animal);

       String json = new Gson().toJson(map);
       System.out.println(json);

       Map jsonToMap = fromJson(json);
       System.out.println(jsonToMap);

       Animal animal1 = jsonToMap.get("cat");
       System.out.println(animal1.getEats());
   }

   public static  T fromJson(String str) {
       return new Gson().fromJson(str, new TypeToken() {
       }.getType());
   }

}

上的代码运行会提示如下异常:

Exception in thread "main" java.lang.ClassCastException: com.google.gson.internal.LinkedTreeMap cannot be cast to com.uaf.rabbitmq.producer.Animal
   at com.uaf.rabbitmq.producer.Test2.main(Test2.java:30)

异常原因主要是这句:new Gson().fromJson(str, new TypeToken() {}.getType());

这句在实际执行的时候,List中的T并未传入实际的泛型参数,导致Gson按照LinkedTreeMap来解析JSON,以致发生了错误;这就是一个在编译期泛型类型擦除所导致的问题;

解决这个问题我们需要修改fromJson方法

public class Test2 {

   public static void main(String[] args) {
       Map map = new HashMap();
       Animal animal = new Animal();
       animal.setVegetarian(true);
       animal.setEats("fish");
       map.put("cat", animal);

       String json = new Gson().toJson(map);
       System.out.println(json);

       Map jsonToMap = fromJson(json,  
       new TypeToken>() {}.getType());
       System.out.println(jsonToMap);

       Animal animal1 = jsonToMap.get("cat");
       System.out.println(animal1.getEats());

   }

   public static  T fromJson(String str, Type type) {
       return new Gson().fromJson(str, type);
   }

}

在Gson中提供了TypeToken解决泛型运行时类型擦除问题,TypeToken 这个类来帮助我们捕获像Map这样的泛型信息。上文创建了一个匿名内部类,这样Java编译器就会把泛型信息编译到这个匿名内部类里,然后在运行时就可以被getType()方法用反射API提取到。

文章来源网络,作者:运维,如若转载,请注明出处:https://shuyeidc.com/wp/222206.html<

(0)
运维的头像运维
上一篇2025-04-15 09:34
下一篇 2025-04-15 09:35

相关推荐

  • 个人主题怎么制作?

    制作个人主题是一个将个人风格、兴趣或专业领域转化为视觉化或结构化内容的过程,无论是用于个人博客、作品集、社交媒体账号还是品牌形象,核心都是围绕“个人特色”展开,以下从定位、内容规划、视觉设计、技术实现四个维度,详细拆解制作个人主题的完整流程,明确主题定位:找到个人特色的核心主题定位是所有工作的起点,需要先回答……

    2025-11-20
    0
  • 社群营销管理关键是什么?

    社群营销的核心在于通过建立有温度、有价值、有归属感的社群,实现用户留存、转化和品牌传播,其管理需贯穿“目标定位-内容运营-用户互动-数据驱动-风险控制”全流程,以下从五个维度展开详细说明:明确社群定位与目标社群管理的首要任务是精准定位,需明确社群的核心价值(如行业交流、产品使用指导、兴趣分享等)、目标用户画像……

    2025-11-20
    0
  • 香港公司网站备案需要什么材料?

    香港公司进行网站备案是一个涉及多部门协调、流程相对严谨的过程,尤其需兼顾中国内地与香港两地的监管要求,由于香港公司注册地与中国内地不同,其网站若主要服务内地用户或使用内地服务器,需根据服务器位置、网站内容性质等,选择对应的备案路径(如工信部ICP备案或公安备案),以下从备案主体资格、流程步骤、材料准备、注意事项……

    2025-11-20
    0
  • 如何企业上云推广

    企业上云已成为数字化转型的核心战略,但推广过程中需结合行业特性、企业痛点与市场需求,构建系统性、多维度的推广体系,以下从市场定位、策略设计、执行落地及效果优化四个维度,详细拆解企业上云推广的实践路径,精准定位:明确目标企业与核心价值企业上云并非“一刀切”的方案,需先锁定目标客户群体,提炼差异化价值主张,客户分层……

    2025-11-20
    0
  • PS设计搜索框的实用技巧有哪些?

    在PS中设计一个美观且功能性的搜索框需要结合创意构思、视觉设计和用户体验考量,以下从设计思路、制作步骤、细节优化及交互预览等方面详细说明,帮助打造符合需求的搜索框,设计前的规划明确使用场景:根据网站或APP的整体风格确定搜索框的调性,例如极简风适合细线条和纯色,科技感适合渐变和发光效果,电商类则可能需要突出搜索……

    2025-11-20
    0

发表回复

您的邮箱地址不会被公开。必填项已用 * 标注