Java--泛型

2021-10-05  本文已影响0人  aruba

JDK5提出了新特性:泛型。它允许我们在不知道变量类型的情况下,传入类型参数,在设计框架时,我们会大量的使用泛型,因为泛型的特性:动态,上下边界,编译检查等,特别适合架构设计

一、泛型上手

1.类属性使用泛型

定义泛型可以使用除关键字外的任意名字(遵循变量名的规则),使用"<泛型名>"来表示你需要使用泛型参数:

public class Data<T> {
    private T data;
}
2.类方法使用泛型

上面的类中,我们可以直接使用泛型:T,来作为非静态方法的入参和返回参数:

public class Data<T> {
    private T data;

    public T getData() {
        return data;
    }

    public void setData(T data) {
        this.data = data;
    }

}

我们也可以直接在方法上定义一个或多个全新的泛型,为了和类定义的T作区别,使用R来定义泛型名:

public class Data<T> {
    private T data;

    public T getData() {
        return data;
    }

    public void setData(T data) {
        this.data = data;
    }

    public <R> R mapperData(T data) {
        return (R) data;
    }
}
3.泛型使用

实例化对象时,可以使用两种方式使用泛型,定义对象类型时使用和实例化时使用:

        Data<String> data = new Data<>();
        Data data1 = new Data<String>();

两者有区别:

        Data<String> data = new Data<>();
        Data data1 = new Data<String>();

        String data2 = data.getData();
        Object data3 = data1.getData();

还可以使用"?"通配符,来表示任意类,其实就是Object:

        Data<?> data4 = new Data<>();
        Object data5 = data4.getData();

二、类型擦除

Java中的泛型为伪泛型,在编译时,会对泛型进行擦除,子类最终使用桥接来调用相应类的方法,类型擦除会导致在类型转换时报出异常
我们使用List时,指定其泛型为String,但是我们仍然可以利用反射,存入其他类型变量:

        List<String> list = new ArrayList<>();
        list.add("123");
        try {
            Method addMethod = list.getClass().getMethod("add", Object.class);
            addMethod.invoke(list, 123);
        } catch (NoSuchMethodException e) {
            e.printStackTrace();
        } catch (InvocationTargetException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        }

        System.out.println(list.size());

结果:
2

反射获取的是类元信息,说明实际Class字节码中的add方法,并没有真正使用我们指定的泛型
当我们使用这个变量时,就会报错了:

        System.out.println(list.get(1));

结果:

Exception in thread "main" java.lang.ClassCastException: java.lang.Integer cannot be cast to java.lang.String
    at GenericTest.main(GenericTest.java:34)

三、上下边界

先把类型擦除放一边,泛型也可以限定其上下边界,即它需要继承至某个父类,或者是它的某个父类

1.上边界

使用extends关键字:

public class Data<T extends Number> {
    private T data;

    public T getData() {
        return data;
    }

    public void setData(T data) {
        this.data = data;
    }

}

这样我们就只能使用Number的子类或Number本身作为泛型了

        Data<Integer> data = new Data<>();
2.下边界

使用super关键字,只能在使用泛型时使用:

Class<? super Integer> clz = Number.class;

四、桥接

当我们自定义一个类继承至一个泛型类,并指定其泛型时,父类的方法会被重载:

public class DataImpl extends Data<Integer>{
    @Override
    public Integer getData() {
        return super.getData();
    }

    @Override
    public void setData(Integer data) {
        super.setData(data);
    }
}

而通过上面的了解,类型擦除会将父类的这两个方法的泛型擦除,在字节码中,使用Object代替。可以看到下面Data类反转义后的内容,其中Field data:Ljava/lang/Object;的注释,表示实际是使用了Object

  public T getData();
    flags: ACC_PUBLIC
    Code:
      stack=1, locals=1, args_size=1
         0: aload_0
         1: getfield      #2                  // Field data:Ljava/lang/Object;
         4: areturn
      LineNumberTable:
        line 5: 0
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
               0       5     0  this   LData;
      LocalVariableTypeTable:
        Start  Length  Slot  Name   Signature
            0       5     0  this   LData<TT;>;
    Signature: #20                          // ()TT;

  public void setData(T);
    flags: ACC_PUBLIC
    Code:
      stack=2, locals=2, args_size=2
         0: aload_0
         1: aload_1
         2: putfield      #2                  // Field data:Ljava/lang/Object;
         5: return
      LineNumberTable:
        line 9: 0
        line 10: 5
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
               0       6     0  this   LData;
               0       6     1  data   Ljava/lang/Object;
      LocalVariableTypeTable:
        Start  Length  Slot  Name   Signature
            0       6     0  this   LData<TT;>;
            0       6     1  data   TT;
    Signature: #23                          // (TT;)V

子类DataImpl反转义后的内容为:

  Last modified 2021-10-5; size 761 bytes
  MD5 checksum b30c50b7d1009fcd65b68b48e39aee97
  Compiled from "DataImpl.java"
public class DataImpl extends Data<java.lang.Integer>
  Signature: #25                          // LData<Ljava/lang/Integer;>;
  SourceFile: "DataImpl.java"
  minor version: 0
  major version: 51
  flags: ACC_PUBLIC, ACC_SUPER
Constant pool:
   #1 = Methodref          #8.#28         //  Data."<init>":()V
   #2 = Methodref          #8.#29         //  Data.getData:()Ljava/lang/Object;
   #3 = Class              #30            //  java/lang/Integer
   #4 = Methodref          #8.#31         //  Data.setData:(Ljava/lang/Object;)V
   #5 = Methodref          #7.#32         //  DataImpl.setData:(Ljava/lang/Integer;)V
   #6 = Methodref          #7.#33         //  DataImpl.getData:()Ljava/lang/Integer;
   #7 = Class              #34            //  DataImpl
   #8 = Class              #35            //  Data
   #9 = Utf8               <init>
  #10 = Utf8               ()V
  #11 = Utf8               Code
  #12 = Utf8               LineNumberTable
  #13 = Utf8               LocalVariableTable
  #14 = Utf8               this
  #15 = Utf8               LDataImpl;
  #16 = Utf8               getData
  #17 = Utf8               ()Ljava/lang/Integer;
  #18 = Utf8               setData
  #19 = Utf8               (Ljava/lang/Integer;)V
  #20 = Utf8               data
  #21 = Utf8               Ljava/lang/Integer;
  #22 = Utf8               (Ljava/lang/Object;)V
  #23 = Utf8               ()Ljava/lang/Object;
  #24 = Utf8               Signature
  #25 = Utf8               LData<Ljava/lang/Integer;>;
  #26 = Utf8               SourceFile
  #27 = Utf8               DataImpl.java
  #28 = NameAndType        #9:#10         //  "<init>":()V
  #29 = NameAndType        #16:#23        //  getData:()Ljava/lang/Object;
  #30 = Utf8               java/lang/Integer
  #31 = NameAndType        #18:#22        //  setData:(Ljava/lang/Object;)V
  #32 = NameAndType        #18:#19        //  setData:(Ljava/lang/Integer;)V
  #33 = NameAndType        #16:#17        //  getData:()Ljava/lang/Integer;
  #34 = Utf8               DataImpl
  #35 = Utf8               Data
{
  public DataImpl();
    flags: ACC_PUBLIC
    Code:
      stack=1, locals=1, args_size=1
         0: aload_0
         1: invokespecial #1                  // Method Data."<init>":()V
         4: return
      LineNumberTable:
        line 1: 0
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
               0       5     0  this   LDataImpl;

  public java.lang.Integer getData();
    flags: ACC_PUBLIC
    Code:
      stack=1, locals=1, args_size=1
         0: aload_0
         1: invokespecial #2                  // Method Data.getData:()Ljava/lang/Object;
         4: checkcast     #3                  // class java/lang/Integer
         7: areturn
      LineNumberTable:
        line 4: 0
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
               0       8     0  this   LDataImpl;

  public void setData(java.lang.Integer);
    flags: ACC_PUBLIC
    Code:
      stack=2, locals=2, args_size=2
         0: aload_0
         1: aload_1
         2: invokespecial #4                  // Method Data.setData:(Ljava/lang/Object;)V
         5: return
      LineNumberTable:
        line 9: 0
        line 10: 5
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
               0       6     0  this   LDataImpl;
               0       6     1  data   Ljava/lang/Integer;

  public void setData(java.lang.Object);
    flags: ACC_PUBLIC, ACC_BRIDGE, ACC_SYNTHETIC
    Code:
      stack=2, locals=2, args_size=2
         0: aload_0
         1: aload_1
         2: checkcast     #3                  // class java/lang/Integer
         5: invokevirtual #5                  // Method setData:(Ljava/lang/Integer;)V
         8: return
      LineNumberTable:
        line 1: 0
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
               0       9     0  this   LDataImpl;

  public java.lang.Object getData();
    flags: ACC_PUBLIC, ACC_BRIDGE, ACC_SYNTHETIC
    Code:
      stack=1, locals=1, args_size=1
         0: aload_0
         1: invokevirtual #6                  // Method getData:()Ljava/lang/Integer;
         4: areturn
      LineNumberTable:
        line 1: 0
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
               0       5     0  this   LDataImpl;
}

以setData方法为例,可以观察发现,含有两个setData方法:

  public void setData(java.lang.Integer);
    flags: ACC_PUBLIC
    Code:
      stack=2, locals=2, args_size=2
         0: aload_0
         1: aload_1
         2: invokespecial #4                  // Method Data.setData:(Ljava/lang/Object;)V
         5: return
      LineNumberTable:
        line 9: 0
        line 10: 5
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
               0       6     0  this   LDataImpl;
               0       6     1  data   Ljava/lang/Integer;

  public void setData(java.lang.Object);
    flags: ACC_PUBLIC, ACC_BRIDGE, ACC_SYNTHETIC
    Code:
      stack=2, locals=2, args_size=2
         0: aload_0
         1: aload_1
         2: checkcast     #3                  // class java/lang/Integer
         5: invokevirtual #5                  // Method setData:(Ljava/lang/Integer;)V
         8: return
      LineNumberTable:
        line 1: 0
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
               0       9     0  this   LDataImpl;

入参为Object的setData方法中,会调用checkcast指令检查类型是否为#3,即是否为Integer类型。然后又执行了invokevirtual #5,查看注释我们也可以知道,最终调用的是入参为Integer类型的setData方法,入参为Object的setData方法就是桥方法,重载只是假象,就是为了解决类型擦除和多态的冲突。这些概念实际就是为了兼容以前没有泛型的JDK版本

上一篇下一篇

猜你喜欢

热点阅读