今天就跟大家聊聊有关Java 泛型在哪些情况下无法使用,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。
我们提供的服务有:成都网站建设、网站制作、微信公众号开发、网站优化、网站认证、广安ssl等。为上千家企事业单位解决了网站和推广的问题。提供周到的售前咨询和贴心的售后服务,是有科学管理、有技术的广安网站制作公司
1. 前言
Java 1.5引入了泛型来保证类型安全,防止在运行时发生类型转换异常,让类型参数化,提高了代码的可读性和重用率。但是有些情况下泛型也是不允许使用的,今天就总结一下编码中不能使用泛型的一些场景。
2. 基本类型无法直接使用泛型
以下写法是错误的:
// error
Map wrong= new HashMap<>()
基本类型是不能够作为泛型类型的,需要使用它们对应的包装类。
// OK
Map wrong= new HashMap<>()
泛型类型可以理解为一个抽象类型,只是代表了类型的抽象,因此我们不能直接实例化它,下面的做法也是错误的:
public E first(List list){
// error
E e = new E();
return list.get(0);
}
Java中的静态类型随着类加载而实例化,此时泛型的具体类型并没有声明。同时因为静态变量作为所有对象的共享变量,只有类实例化或者方法调用时才能确定其类型。如果是泛型类型将无法确定其类型。同样在类上声明的泛型也无法作为返回值类型出现在类的静态方法中,下面的写法也是错误的:
public class Generic{
// 不能将类声明的泛型类型作为静态变量
public static T t;
// 也不能将类声明的泛型类型作为 静态方法的返回值
public static T rtval(List list){
return list.get(0);
}
}
Java中的泛型是伪泛型,在编译期会被擦除,运行的字节码中不存在泛型,所以下面的判断条件无法进行:
public static void wrong(List list) {
// error
if (list instanceof ArrayList) {
}
}
但是泛型的无界通配符
>
可以进行instanceof
判断,你仔细想想为什么。
首先下面这种写法是对的:
// 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();
// 这显然不合理
下面的两种写法将引发编译错误:
// 不能间接地扩展 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 {
}
}
由于泛型擦除的原因,以下的不视为方法的重载且无法编译 :
public class NoReload {
public void sets(Set strSet) { }
public void sets(Set intSet) { }
}
看完上述内容,你们对Java 泛型在哪些情况下无法使用有进一步的了解吗?如果还想了解更多知识或者相关内容,请关注创新互联行业资讯频道,感谢大家的支持。