Java SE 进阶复习(五)泛型

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接: https://blog.csdn.net/Unicorn_JF/article/details/100069882

1、Java中的泛型是什么 ? 使用泛型的好处是什么?

泛型是Java SE 1.5的新特性,泛型的本质是参数化类型,也就是说所操作的数据类型被指定为一个参数。

好处

  1. 类型安全,提供编译期间的类型检测;
  2. 前后兼容;
  3. 泛化代码,代码可以更多的重复利用;
  4. 性能较高,用GJ(泛型JAVA)编写的代码可以为java编译器和虚拟机带来更多的类型信息,这些信息对java程序做进一步优化提供条件。

2、什么是类型擦除 ?泛型如何工作?

类型检查:在生成字节码之前提供类型检查。
类型擦除:所有类型参数都用他们的限定类型替换,包括类、变量和方法(类型擦除)。

如果类型擦除和多态性发生了冲突时,则在子类中生成桥方法解决。

如果调用泛型方法的返回类型被擦除,则在调用该方法时插入强制类型转换。

工作原理:泛型是通过类型擦除来实现的。

编译器在编译时擦除了所有类型相关的信息,所以在运行时不存在任何类型相关的信息。

例如 List在运行时仅用一个List来表示。这样做的目的,是确保能和Java 5之前的版本开发二进制类库进行兼容。你无法在运行时访问到类型参数,因为编译器已经把泛型类型转换成了原始类型。

3、可以把List传递给一个接受List参数的方法吗?

对任何一个不太熟悉泛型的人来说,这个Java泛型题目看起来令人疑惑,因为乍看起来String是一种Object,所以 List应当可以用在需要List的地方,但是事实并非如此。

真这样做的话会导致编译错误。如果你再深一步考虑,你会发现Java这样做是有意义的,因为List可以存储任何类型的对象包括String, Integer等等,而List却只能用来存储String s。

4、如何阻止Java中的类型未检查的警告?

如果你把泛型和原始类型混合起来使用,例如下列代码,java 5的javac编译器会产生类型未检查的警告,例如

List<String> rawList = newArrayList()

注意: Hello.java使用了未检查或称为不安全的操作;

这种警告可以使用@SuppressWarnings(“unchecked”)注解来屏蔽。

5、Java中List和原始类型List之间的区别?

原始类型带参数类型之间的主要区别是:

  1. 在编译时编译器不会对原始类型进行类型安全检查,却会对带参数的类型进行检查,通过使用Object作为类型,可以告知编译器该方法可以接受任何类型的对象,比如String或Integer。
  2. 可以把任何带参数的类型传递给原始类型List,但却不能把List传递给接受 List的方法,因为会产生编译错误。

6、Array中可以用泛型吗?

不能。在实际编程过程中建议使用List来代替Array,因为List可以提供编译期的类型安全保证,而Array却不能。

7、C++模板和java泛型之间有何不同?

java泛型实现根植于“类型消除”这一概念。当源代码被转换为Java虚拟机字节码时,这种技术会消除参数化类型。有了Java泛型,我们可以做的事情也并没有真正改变多少;他只是让代码变得漂亮些。鉴于此,Java泛型有时也被称为“语法糖”。

这和 C++模板截然不同。在 C++中,模板本质上就是一套宏指令集,只是换了个名头,编译器会针对每种类型创建一份模板代码的副本。

由于架构设计上的差异,Java泛型和C++模板有很多不同点:

  1. C++模板可以使用int等基本数据类型。Java则不行,必须转而使用Integer。
  2. 在Java中,可以将模板的参数类型限定为某种特定类型。
  3. 在C++中,类型参数可以实例化,但java不支持。
  4. 在Java中,类型参数不能用于静态方法(?)和变量,因为它们会被不同类型参数指定的实例共享。在C++,这些类时不同的,因此类型参数可以用于静态方法和静态变量。
  5. 在Java中,不管类型参数是什么,所有的实例变量都是同一类型。类型参数会在运行时被抹去。在C++中,类型参数不同,实例变量也不同。

猜你喜欢

转载自blog.csdn.net/Unicorn_JF/article/details/100069882
今日推荐