在java中,普通的类和方法只能用具体的类型,这对代码的限制很大,代码的可重用性大大降低。

那么如何才能让同一个类和方法使用不同类型的对象呢?在接触泛型之前我们可能会想到通过类型转换的方法来实现。

  1. public class Test {
  2. Object o;
  3.  
  4. public Test(Object o) {
  5. this.o = o;
  6. }
  7.  
  8. public void set(Object o){
  9. this.o=o;
  10. }
  11.  
  12. public Object get(){
  13. return o;
  14. }
  15.  
  16. public static void main(String[] args) {
  17. String str="hello world";
  18. Test test=new Test(str);
  19. String str2=(String) test.get();
  20. }
  21.  
  22. }

但是这种方法有很大的缺陷,容易在类型转换过程中出现错误,而使用泛型能很好地避免这个错误的出现。


泛型能够能够指定容器容器要装什么类型的对象,无须类型转换。

泛型类:

  1. public class Test<T> {
  2. private T a;
  3.  
  4. public Test(T a) {
  5. this.a = a;
  6. }
  7.  
  8. public T getA() {
  9. return a;
  10. }
  11.  
  12. public void setA(T a) {
  13. this.a = a;
  14. }
  15.  
  16. public static void main(String[] args) {
  17. Test<String> test = new Test<>("test");
  18. System.out.println(test.getA());
  19. }
  20. }

泛型方法:

  1. public <T> String getClassName(T t) {
  2. return t.getClass().getSimpleName();
  3. }

对于泛型方法和泛型类,我们通常首先选择泛型方法,毕竟,泛型类的束缚更大,它表示我们在整个类中都要使用着这类型;相对而言泛型方法就灵活许多,我们可以在不同的方法中使用不同的类型。


泛型擦除:

  1. public static void main(String[] args) {
  2. List<Integer> list1 = new ArrayList<>();
  3. List<String> list2 = new ArrayList<>();
  4. System.out.println(list1.getClass()==list2.getClass());
  5. /*
  6. output:
  7. true
  8. */
  9. }

list1和list2的泛型参数类型不相同,但从上面的结果中不难看出,系统认为它们是相同的类型,因为泛型只在编译过程起作用,当程序运行时泛型参数的具体信息都被擦除。


边界:

  1. class Animal{
  2. void eat(){}
  3. }
  4.  
  5. class Bird extends Animal {
  6. }

当我们已经确认此泛型参数为一固定类型Animal或Animal的子类时,我们希望能够调用Animal对象中的eat方法, 那我们又该如何实现呢?

这里我们需要用到泛型边界,它通过extends关键字来实现:

  1. public class Test<T extends Animal> {
  2. public void test(T t) {
  3. t.eat();
  4. }
  5. }

这里声明了边界Animal,它可以调用Animal中的成员。如果没有声明边界,则默认边界为Object。

当然如果只是实现这个目的,我们无需使用泛型:

  1. public class Test{
  2. public void test(Animal a){
  3. a.eat();
  4. }
  5. }

好像这里使用泛型没什么用处,其实并非如此。当泛型方法有泛型类型的返回值时,它能返回确切的类型:

  1. public class Test<T extends Animal> {
  2. public T get(T t) {
  3. t.eat();
  4. return t;
  5. }
  6.  
  7. public static void main(String[] args) {
  8. Animal a = new Test<>().get(new Animal());
  9. Bird b = new Test<Bird>().get(new Bird());
  10. }
  11. }

这比不使用泛型灵活许多,所以,我们在考虑是否使用泛型时要根据具体情况而定,看看结构是否足够复杂,使用泛型是否会给我们带来方便。


通配符:

我们再来看一个错误例子:

  1. List<Animal> list=new ArrayList<Bird>(); //error:Type mismatch: cannot convert from ArrayList<Bird> to List<Animal>

想当初第一次运行类似的代码时,我有些无法理解,Bird是Animal的子类,为何就不能向上转型呢呢,这泛型也太呆板了。仔细一想才发现,这根本就不是向上转型,List<Animal>代表Animal类型的列表,而ArrayList<Bird>代表Bird类型的列表。

可是我们就是想要实现这样的功能,那该怎么做?这里就需要用到通配符:

  1. List<? extends Animal> list=new ArrayList<Bird>();

泛型与数组:

我们无法直接创建泛型数组实例,因为你数组类型必须是确定的,以此确保类型安全,但是擦除却使类型参数信息消失。

  1. List<String>[] lists=new ArrayList<String>[12]; //error

但是们可以声明一个泛型数组:

  1. List<String>[] lists;

根据以上两个特性,我们可以创建一个非泛型数组,再进行转型,从而获得泛型数组:

  1. List<String>[] lists=(List<String>[])new ArrayList[12];
  2. //或简化为:List<String>[] lists=new ArrayList[12];

java之泛型的使用的更多相关文章

  1. [改善Java代码]Java的泛型是类型擦除的

    泛型可以减少强制类型的转换,可规范集合的元素类型,还可以提高代码的安全性和可读性,正是因为有了这些优点,自从Java引入泛型之后,项目的编码规则上便多了一条,优先使用泛型. Java泛型(Generi ...

  2. Java 中泛型的全面解析(转)

    Java泛型(generics) 是JDK 5中引入的一个新特性,允许在定义类和接口的时候使用类型参数(type parameter).声明的类型参数在使用时用具体的类型来替换.泛型最主要的应用是在J ...

  3. Java中泛型 类型擦除

    转自:Java中泛型是类型擦除的 Java 泛型(Generic)的引入加强了参数类型的安全性,减少了类型的转换,但有一点需要注意:Java 的泛型在编译器有效,在运行期被删除,也就是说所有泛型参数类 ...

  4. Java 泛型 Java使用泛型的意义

    Java 泛型 Java使用泛型的意义 @author ixenos 直接意义 在编译时保证类型安全 根本意义 a) 类型安全问题源自可复用性代码的设计,泛型保证了类型安全的复用模板 b) 使用复用性 ...

  5. 跟着刚哥梳理java知识点——泛型(十三)

    一. 泛型概念的提出(为什么需要泛型)? 首先,我们看下下面这段简短的代码: public class GenericTest { public static void main(String[] a ...

  6. 【Java】泛型学习笔记

    参考书籍 <Java核心技术:卷1> 泛型, 先睹为快 先通过一个简单的例子说明下Java中泛型的用法: 泛型的基本形式类似于模板, 通过一个类型参数T, 你可以"私人定制&qu ...

  7. [转] Java 的泛型擦除和运行时泛型信息获取

    原文链接 https://my.oschina.net/lifany/blog/875769 前言 现在很多程序员都会在简历中写上精通 Java.但究竟怎样才算是精通 Java 呢?我觉得不仅要熟练掌 ...

  8. Java 容器 & 泛型:五、HashMap 和 TreeMap的自白

    Writer:BYSocket(泥沙砖瓦浆木匠) 微博:BYSocket 豆瓣:BYSocket Java 容器的文章这次应该是最后一篇了:Java 容器 系列. 今天泥瓦匠聊下 Maps. 一.Ma ...

  9. 【译】9. Java反射——泛型

    原文地址:http://tutorials.jenkov.com/java-reflection/generics.html ===================================== ...

  10. Java“禁止”泛型数组

    Java“禁止”泛型数组 原文:https://blog.csdn.net/yi_Afly/article/details/52058708 1. 泛型定义泛型编程是一种通过参数化的方式将数据处理与数 ...

随机推荐

  1. T1215拯救公主

    1 #include <cstdio> 2 #include <queue> 3 #include <set> 4 #include <cstring> ...

  2. java网络通信不止UDP,TCP

    预备知识 多线程 实现多线程 线程池 IO流 核心功能就是读和写 扩展功能对什么读写,怎么读写,如何优化读写 网络基础 IP IP规定网络上所有的设备都必须有一个独一无二的IP地址,就好比是邮件上都必 ...

  3. Redis 通过 RDB 方式进行数据备份与还原

    Redis 通过 RDB 方式进行数据备份与还原 Intro 有的时候我们需要对 Redis 的数据进行迁移,今天介绍一下通过 RDB(快照)文件进行 Redis 数据的备份和还原 Redis 持久化 ...

  4. WERTYU_键盘错位(JAVA语言)

    package 第三章; import java.util.Scanner; /*  *  把手放在键盘上时,稍不注意就会往右错一位.这样,输入Q会变成输入W,输入J会变成输入K等.        输 ...

  5. 【秒懂音视频开发】12_播放WAV

    对于WAV文件来说,可以直接使用ffplay命令播放,而且不用像PCM那样增加额外的参数.因为WAV的文件头中已经包含了相关的音频参数信息. ffplay in.wav 接下来演示一下如何使用SDL播 ...

  6. MySQL5.7和MySQL8.0通用配置文件

    MySQL5.7 my.cnf配置 [client] port=3306 socket=/log/mysql/mysql.sock [mysql] socket=/log/mysql/mysql.so ...

  7. 【Java】 5.0 判断与转换

    [概述] 在if/条件语句中,我们已经谈及判断了,这次将详细讲讲一些逻辑判断 基本逻辑 &:且,And,需要二者均为True |:或,Or ,需要二者其一为False即可 ^:异或,XOE,两 ...

  8. OO第三单元——基于JML的社交网络总结

    OO第三单元--基于JML的社交网络总结 一.JML知识梳理 1)JML的语言基础以及基本语法 JML是用于java程序进行规格化设计的一种表示语言,是一种行为接口规格语言.其为严格的程序设计提供了一 ...

  9. SQLlite实现增删查改

    activity_main.xml文件: <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android ...

  10. 《青橙商城》day01避坑指南

    1.persistence-api报红 问题: 在模块qingcheng_pojo下的pom.xml中,报红 <dependencies> <dependency> <g ...