Java 泛型详解

作者: MrTrying | 来源:发表于2017-07-09 23:18 被阅读188次

一、定义

Java泛型是JDK5中引入的新特性,提供在编译时类型安全监测机制,本质是参数化类型(即所操作的数据类型被指定为一个参数)

    ArrayList list = new ArrayList();
    list.add("string");
    list.add(true);
    list.add(123);
    
    for(int index = 0 ; index < list.size() ; index++){
        String str = list.get(index);
        Log.i("tzy","str = " + str);
    }

在没有泛型约束的情况下,写代码的时候ArrayList可以添加任何对象(true和123虽然是基础类型,但是有对应的包装类,可以进行自动打包和解包),在for循环中的第一句代码就会报ClassCastException异常了,因为集合中第二个对象是Boolean对象,而且这个异常在运行的时候才会暴露出来,看一段改进的代码

    ArrayList<String> list = new ArrayList<>();
    list.add("string");
    //以下注释的代码在编写代码就会报错提示
    //list.add(true);
    //list.add(123);
    
    for(int index = 0 ; index < list.size() ; index++){
        String str = list.get(index);
        Log.i("tzy","str = " + str);
    }

添加泛型以后的代码,无法在编译时的状态向集合中添加不符合泛型类型的对象,上面一段简单代码就可以看出泛型意义了

泛型只能作用于编译时,对反射(运行时)是无效的。
泛型只支持Object的类型,而对于8大基础数据类型(byteshortintlongfloatdoublecharboolean)不支持,所以相应的有8个封装类(ByteShortIntegerLongFloatDoubleCharacterBoolean)提供给配合泛型使用,在反省中使用基础类型的时候,封装类有自动装箱/拆箱的功能,为我们省去了麻烦。

二、泛型使用

1、自定义泛型方法

可以写一个方法,调用该方法的时候可以传入不同类型的参数

规则:

  • 声明泛型方法的类型参数声明部分,在方法返回类型之前
  • 每个类型参数和僧名部分包含一个或者多个类型参数,参数中间用逗号隔开
  • 类型参数可以用来声明返回值类型,并且作为泛型方法得到的实际参数类型的占位符
public <E> void printArray(E[] array){
    for(E elment:array){
        Log.i("tzy","elment = " + elment);
    }
}

public void test(){
    Integer[] intArray = [1,2,3,4];
    Character[] charArray = ['a','b','c','d'];
    String[] stringArray = ["x","y","z","."];
    
    printArray(intArray);
    printArray(charArray);
    printArray(stringArray);
}

2、自定义泛型类

泛型类的声明和非泛型类的声明类似,在类名后面添加了类型参数声明部分(数量和写法也是一样的)。一个泛型参数(又称类型变量),适用于

class Computer{
    public String toString(){
        return getClass().getSimpleName();
    }
}

class MacBookPro extends Computer{}
class Alienware extends Computer{}
class Thinkpad extends Computer{}

/**使用泛型的类*/
public class Persion<? extends Computer>{
    private Computer mComputer;
    
    public void setComputer(Computer computer){
        this.mComputer = computer;
    }

    public Computer getComputer(){
        return this.mComputer;
    }
}
public void test(){
    Persion andy = new Persion<MacBookPro>();
    Persion peter = new Persion<Alienware>();

    andy.setComputer(new MacBookPro());
    peter.setComputer(new Alienware());

    Log.i("tzy","andy's computer is " + andy.getComputer().toString());
    Log.i("tzy","peter's computer is" + peter.getComputer().toString());
}

3、自定义泛型类接口

泛型接口与泛型类非常相似,只不过定义为interface

public interface Computer<T>{
    public T getCPU();
}

/**定义实现泛型接口的类*/
class MacBookPro<String> implements Computer<String>{
    private String cpu;
    
    public MacBookPro(T cpu){
        this.cpu = cpu;
    }

    @Override
    public String getCPU(){
        return cpu;
    }
}

三、类型通配符

类型通配符一般是使用?代替具体的类型参数。
例如 List<?> 在逻辑上是List<String>,List<Integer> 等所有List<具体类型实参>的父类。

public void printData(List<?> data){
    Log.i("tzy","data = " + data.toString());
}

//类型通配符上限通过形如List来定义,如此定义就是通配符泛型值接受Number及其下层子类类型。
public void printData(List<? extends Computer> data){
    Log.i("tzy","data = " + data.toString());
}

//类型通配符下限通过形如 List<? super Number>来定义,表示类型只能接受Number及其三层父类类型
public void printData(List<? spuer Computer> data){
    Log.i("tzy","data = " + data.toString());
}

<? extends T>表示该通配符所代表的类型是T类型的子类。
<? super T>表示该通配符所代表的类型是T类型的父类,只能接受T及其三层父类类型

四、类型限界

类型限界再见括号内置顶,它指定参数类型必须具有的性质。不太明白接着往下看,假如现在我想要编写一个findBest的方法,代码如下

public interface Comparable<T>{
    boolean compareTo(T other);//接口的中方法自动属于public方法
}

public <Computer> Computer findBest(Computer[] arr){
    int bestIndex = 0;
    
    for(int index = 0 ; index < arr.length ; index ++){
        if(arr[i].compareTo(bestIndex))
            bestIndex = i;
    }

    return arr[bestIndex];
}

我们刚开始是这么想的,再来看一遍代码,只有在Computer保证有compareTo()情况下才能不抛异常,改变一下代码public <Computer extends Comparable> Computer,这样程序不会有问题了,但是更好的写法public <Computer extends Comparable<Computer >> Computer,现在就是最优的方案了么?
假设Computer实现Comparable<Computer>,设Alineware继承Computer。此时我们可以知道Alineware实现Comparable<Computer>,这说明Alineware符合Comparable<Computer>的类型要求,ComputerAlineware父类,由此可见,我们不需要知道准确的知道Comparable<T>中的T是什么,因此可以使用通配符,最后的结果就变成了public <Computer extends Comparable<? extends Computer>>

public <Computer extends Comparable<? extends Computer>> Computer findBest(Computer[] arr){
    int bestIndex = 0;
    
    for(int index = 0 ; index < arr.length ; index ++){
        if(arr[i].compareTo(bestIndex))
            bestIndex = i;
    }

    return arr[bestIndex];
}

五、类型擦除

由于虚拟机中没有泛型,只有普通类和普通方法,所有泛型类的类型参数在编译时都会被擦除,即将所有的泛型参数用最顶级的父类替换并移除所有类型参数。(这也是反射可以绕过泛型的原因)

带来的问题:

  • 无法使用带有泛型的形参来实现多态
  • 无法catch不同泛型的同一个异常类
  • 带有泛型的静太类变量是共享的

六、注意

  • static修饰的方法或者域,是不可以引用类的类型变量,因为在类型擦除后类型变量就不存在了。另外,由于实际上只存在一个原始的类,因此static域在该类的泛型实例之间都是共享的
  • 泛型对象无法直接创建,new T();是非法的。T由它的限界代替,这可能是Object(或甚至是抽象类),所以对new的调用没有意义
  • 泛型数组对象也是不能创建的,由于类型擦除,对T[]的类型转换将无法进行

相关文章

  • 泛型,复习

    Java泛型详解:和Class的使用。泛型类,泛型方法的详细使用实例 - LZJWXJ树袋熊 - CSDN博客

  • java 泛型

    java 泛型详解-绝对是对泛型方法讲解最详细的,没有之一

  • 泛型

    java 泛型详解-绝对是对泛型方法讲解最详细的,没有之一Java泛型深入理解加泛型面试数组的协变性与范型的不可变性

  • 详解Java泛型之4——一个例子理解泛型带来的好处

    前面我介绍了关于泛型、通配符以及泛型擦除的相关知识点,大家可以参考以下文章: 详解Java泛型之1——入门泛型必懂...

  • 详解Java泛型之3——十分钟理解泛型擦除

    前面我们介绍了泛型以及通配符的基础概念,可以参考文章: 详解Java泛型之1——入门泛型必懂的知识点[https:...

  • java泛型详解

    java泛型详解 1. 概述 泛型在java中有很重要的地位,在面向对象编程及各种设计模式中有非常广泛的应用。 什...

  • Java泛型、枚举类型详解

    建议看以下博客文章,比较详细且易懂 Java泛型详解 java 枚举(enum) 全面解读

  • Java泛型教程

    Java泛型教程导航 Java 泛型概述 Java泛型环境设置 Java泛型通用类 Java泛型类型参数命名约定 ...

  • 2018-07-16

    Java 阅读 大白话说Java泛型 深入理解Java集合框架 java构造代码块详解 Java中Overload...

  • 泛型程序设计

    诚心推荐这一篇文章更好 java 泛型详解 泛型程序设计 泛型意味着编写的代码可以被不同类型重用 ArrayLis...

网友评论

    本文标题:Java 泛型详解

    本文链接:https://www.haomeiwen.com/subject/xbwehxtx.html