首页
学习
活动
专区
工具
TVP
发布
精选内容/技术社群/优惠产品,尽在小程序
立即前往

Java 泛型基本用法与类型擦除

隆重介绍本文作者:然则

目前就职于百度安全事业部工作,主要负责Android客户端以及Linux平台下的开发,实现 AI 产品的落地,喜欢钻研各种技术,对前后端技术均感兴趣。

简介

Java 在 1.5 引入了泛型机制,泛型本质是参数化类型,也就是说变量的类型是一个参数,在使用时再指定为具体类型。泛型可以用于类、接口、方法,通过使用泛型可以使代码更简单、安全。然而 Java 中的泛型使用了类型擦除,所以只是伪泛型。这篇文章对泛型的使用以及存在的问题做个总结,主要参考自 《Java 编程思想》。

基本用法

泛型类

如果有一个类 用于包装一个变量,这个变量的类型可能是任意的,怎么编写 呢?在没有泛型之前可以这样:

在 中,有一个用 引用的变量。因为任何类型都可以向上转型为 ,所以这个 可以接受任何类型。在取出的时候 只知道它保存的是一个 对象,所以要强制转换为对应的类型。在 方法中, 先是保存了一个字符串,也就是 对象,接着又变为保存一个 对象(参数 1 会自动装箱)。从 中取出变量时强制转换已经比较麻烦,这里还要记住不同的类型,要是转错了就会出现运行时异常。

下面看看 的泛型版本:

在 中, 变量 是一个参数化类型 , 只是一个标识,用其它字母也是可以的。创建 对象的时候,在尖括号中传入了参数 的类型,那么在这个对象中,所有出现 的地方相当于都用 替换了。现在的 的取出来的不是 ,而是 对象,因此不需要类型转换。另外,当调用 时,只能传入 类型,否则编译无法通过。这就保证了 中的类型安全,避免由于不小心传入错误的类型。

通过上面的例子可以看出泛使得代码更简便、安全。引入泛型之后, 库的一些类,比如常用的容器类也被改写为支持泛型,我们使用的时候都会传入参数类型,如:。

泛型方法

泛型不仅可以针对类,还可以单独使某个方法是泛型的,举个例子:

类本身不是泛型的,创建它的对象的时候不需要传入泛型参数,但是它的方法 是泛型方法。在返回类型之前是它的参数标识 ,注意这里有两个泛型参数,所以泛型参数可以有多个。

调用泛型方法时可以不显式传入泛型参数,上面的调用就没有。这是因为编译器会使用参数类型推断,根据传入的实参的类型 (这里是 和 ) 推断出 和 的类型。

类型擦除

什么是类型擦除

的泛型使用了类型擦除机制,这个引来了很大的争议,以至于 的泛型功能受到限制,只能说是”伪泛型“。什么叫类型擦除呢?简单的说就是,类型参数只存在于编译期,在运行时, 的虚拟机 ( ) 并不知道泛型的存在。先看个例子:

上面的代码有两个不同的 和 。在我们看来它们的参数化类型不同,一个保存整性,一个保存字符串。但是通过比较它们的 对象,上面的代码输出是 。这说明在 看来它们是同一个类。而在 、 这些支持真泛型的语言中,它们就是不同的类。

泛型参数会擦除到它的第一个边界,比如说上面的 类,参数类型是一个单独的 ,那么就擦除到 ,相当于所有出现 的地方都用 替换。所以在 看来,保存的变量 还是 类型。之所以取出来自动就是我们传入的参数类型,这是因为编译器在编译生成的字节码文件中插入了类型转换的代码,不需要我们手动转型了。如果参数类型有边界那么就擦除到它的第一个边界,这个下一节再说。

擦除带来的问题

擦除会出现一些问题,下面是一个例子:

上面的 是一个泛型类,内部用一个泛型化的变量 ,在 方法中,调用了 的方法 ,但是这行代码无法编译。因为类型擦除,编译器不确定 是否有 方法。解决这个问题的方法是给 一个边界:

现在 的类型是 ,这表示 必须是 或者 的导出类型。这样,调用 方法才安全。 就是 的边界,因此通过类型擦除后,所有出现 的

地方都用 替换。这样编译器就知道 是有方法 的。

但是这样就抵消了泛型带来的好处,上面的类完全可以改成这样:

所以泛型只有在比较复杂的类中才体现出作用。但是像 这种形式的东西不是完全没有意义的。如果类中有一个返回 类型的方法,泛型就有用了,因为这样会返回准确类型。比如下面的例子:

这里的 方法返回的是泛型参数的准确类型,而不是 。

类型擦除的补偿

类型擦除导致泛型丧失了一些功能,任何在运行期需要知道确切类型的代码都无法工作。比如下面的例子:

通过 创建对象是不行的,一是由于类型擦除,二是由于编译器不知道 是否有默认的构造器。一种解决的办法是传递一个工厂对象并且通过它创建新的实例。

另一种解决的方法是利用模板设计模式:

具体类型的创建放到了子类继承父类时,在 方法中创建实际的类型并返回。

总结

本文介绍了 泛型的使用,以及类型擦除相关的问题。一般情况下泛型的使用比较简单,但是某些情况下,尤其是自己编写使用泛型的类或者方法时要注意类型擦除的问题。接下来会介绍数组与泛型的关系以及通配符的使用,有兴趣的读者可期待下一篇:Java 泛型(二):泛型与数组。

  • 发表于:
  • 原文链接http://kuaibao.qq.com/s/20180329G1PML300?refer=cp_1026
  • 腾讯「腾讯云开发者社区」是腾讯内容开放平台帐号(企鹅号)传播渠道之一,根据《腾讯内容开放平台服务协议》转载发布内容。
  • 如有侵权,请联系 cloudcommunity@tencent.com 删除。

扫码

添加站长 进交流群

领取专属 10元无门槛券

私享最新 技术干货

扫码加入开发者社群
领券