什么情况下不能使用 Java 泛型

简介: 1. 前言Java 1.5 引入了泛型来保证类型安全,防止在运行时发生类型转换异常,让类型参数化,提高了代码的可读性和重用率。但是有些情况下泛型也是不允许使用的,今天就总结一下编码中不能使用泛型的一些场景。2. 基本类型无法直接使用泛型以下写法是错误的:// error Map<int,char> wrong= new HashMap<>()复制代码基本类型是不能够作为泛型类型的,需要使用它们对应的包装类。// OKMap<Integer,Character> wrong= new HashMap<>()复制代码3. 泛型类型无法被直接实例化泛型类型可以理解为一个
  1. 前言

Java 1.5 引入了泛型来保证类型安全,防止在运行时发生类型转换异常,让类型参数化,提高了代码的可读性和重用率。但是有些情况下泛型也是不允许使用的,今天就总结一下编码中不能使用泛型的一些场景。

  1. 基本类型无法直接使用泛型

以下写法是错误的:

// error
Map<int,char> wrong= new HashMap<>()
复制代码
基本类型是不能够作为泛型类型的,需要使用它们对应的包装类。

// OK
Map<Integer,Character> wrong= new HashMap<>()
复制代码

  1. 泛型类型无法被直接实例化

泛型类型可以理解为一个抽象类型,只是代表了类型的抽象,因此我们不能直接实例化它,下面的做法也是错误的:

public E first(List list){

 // error 
    E e = new E();
   return list.get(0);   

}
复制代码

  1. 泛型无法作为静态变量类型

Java 中的静态类型随着类加载而实例化,此时泛型的具体类型并没有声明。同时因为静态变量作为所有对象的共享变量,只有类实例化或者方法调用时才能确定其类型。如果是手游卖号泛型类型将无法确定其类型。同样在类上声明的泛型也无法作为返回值类型出现在类的静态方法中,下面的写法也是错误的:

public class Generic{

// 不能将类声明的泛型类型作为静态变量
public static T t;
// 也不能将类声明的泛型类型作为 静态方法的返回值
public  static  T  rtval(List<T> list){
    return list.get(0);
}

}
复制代码

  1. 无法进行 instanceof 判断

Java 中的泛型是伪泛型,在编译期会被擦除,运行的字节码中不存在泛型,所以下面的判断条件无法进行:

public static void wrong(List list) {

// error 
if (list instanceof ArrayList<Integer>) {   
}

}
复制代码
但是泛型的无界通配符 <?> 可以进行 instanceof 判断,你仔细想想为什么。

  1. 无法创建参数化类型的数组

首先下面这种写法是对的:

// OK
List[] arrayOfLists = new List[2];
复制代码
但是加上了泛型就编译不通过了:

//error
List[] arrayOfLists = new List[2];
复制代码
如果不这么规定将引发以下逻辑错误:

// 如果上面的成立,则下面的也应该成立
Object[] stringLists = new List[];
// 那么我们可以放入 字符串 List
stringLists[0] = new ArrayList();
// 放入 Integer list
stringLists[1] = new ArrayList();
// 这显然不合理
复制代码

  1. 不能直接或者间接扩展Throwable

下面的两种写法将引发编译错误:

// 不能间接地扩展 Throwable
class IndirectException extends Exception {}

// 不能直接地扩展 Throwable
class DirectException extends Throwable {}
复制代码
如果成立将出现:

try {

    // ...
} catch (T e) {   
    // 类型不确定  无法处理具体的异常逻辑
}

复制代码
你如何才能对异常进行具体的处理,这显然不便于精确的异常处理逻辑。但是你可以抛出一个 不确定的异常,但是同样不能在静态方法中使用类声明的泛型:

class Parser {
// 这样是对的

public void okThrow(File file) throws T {      
    // ...
}
// 静态方法不能出现类声明的泛型类型作为返回值和异常
public static void wrongThrow(File file) throws T {      
}

}
复制代码

  1. 泛型擦除后相同参数签名的方法不能重载

由于泛型擦除的原因,以下的不视为方法的重载且无法编译 :

public class NoReload {

public void sets(Set<String> strSet) { }
public void sets(Set<Integer> intSet) { }

}
复制代码

  1. 总结

今天总结了 Java 泛型的一些使用误区,虽然平常 IDE 的提示会告诉我们,但是这也是我们经常会忽略的一些知识点。如果有不足之处,请留言指正。如果你想对泛型了解更多,可关注公众号:码农小胖哥 回复 generic 获取相关的学习笔记。

目录
相关文章
|
4天前
|
存储 Java 编译器
深入理解 Java 泛型和类型擦除
【4月更文挑战第19天】Java泛型是参数化类型,增强安全性与可读性,但存在类型擦除机制。类型擦除保证与旧版本兼容,优化性能,但也导致运行时无法访问泛型信息、类型匹配问题及数组创建限制。为应对这些问题,可使用Object类、instanceof运算符,或借助Guava库的TypeToken获取运行时类型信息。
|
4天前
|
JavaScript Java 编译器
Java包装类和泛型的知识点详解
Java包装类和泛型的知识点的深度理解
|
4天前
|
Java
java中的泛型类型擦除
java中的泛型类型擦除
14 2
|
4天前
|
安全 Java 程序员
Java 泛型
Java 泛型
15 0
|
4天前
|
安全 Java 编译器
java泛型浅谈
java泛型浅谈
6 1
|
4天前
|
存储 安全 Java
掌握8条泛型规则,打造优雅通用的Java代码
掌握8条泛型规则,打造优雅通用的Java代码
掌握8条泛型规则,打造优雅通用的Java代码
|
4天前
|
Java
JAVA难点包括异常处理、多线程、泛型和反射,以及复杂的分布式系统知识
【5月更文挑战第2天】JAVA难点包括异常处理、多线程、泛型和反射,以及复杂的分布式系统知识。入坑JAVA因它的面向对象特性、平台无关性、强大的标准库和活跃的社区支持。
42 2
|
4天前
|
安全 Java 编译器
【JAVA】泛型和Object的区别
【JAVA】泛型和Object的区别
|
4天前
|
存储 算法 Java
滚雪球学Java(20):Java泛型与枚举:提升代码灵活性与可读性
【4月更文挑战第9天】🏆本文收录于「滚雪球学Java」专栏,专业攻坚指数级提升,希望能够助你一臂之力,帮你早日登顶实现财富自由🚀;同时,欢迎大家关注&&收藏&&订阅!持续更新中,up!up!up!!
30 1
滚雪球学Java(20):Java泛型与枚举:提升代码灵活性与可读性
|
4天前
|
Java