java泛型的详解

泛型

泛型类派生子类

  • 子类是泛型类,那么子类和父类的泛型类型要一致,也就是子类的泛型中必须存在父类的泛型,也就是子类的泛型可以进行扩展,子类的泛型数量多余父类。

    • public class Child<T> extends Father<T>{
          @Override
          public T getMoney() {
              return super.getMoney();
          }
      }
      
    • public class Child<T,E,K> extends Father<T>{
          @Override
          public T getMoney() {
              return super.getMoney();
          }
      }
      
  • 子类不是泛型类,父类就要明确泛型的数据类型,不能再用T,E,K等字母,因为这是没有明确的。

    • 父类不写泛型,默认为Object

      public class Child extends Father{
          @Override
          public Object getMoney() {
              return super.getMoney();
          }
      }
      
    • 父类写了明确的泛型,那么子类的类型会和他一样

      public class Child extends Father<String>{
          @Override
          public String getMoney() {
              return super.getMoney();
          }
      }
      

泛型接口

  • 泛型接口语法定义

    interface 接口名称 <泛型标识,泛型标识......> {
      泛型标识 方法名();
    }
    
  • 泛型接口的使用(和泛型类的派生类相似)

    • 实现类不是泛型类,接口要明确数据类型。

      public class son implements Father{
      
          @Override
          public Object getKey() {
              return null;
          }
      }
      
      public class son implements Father<String>{
      
          @Override
          public String getKey() {
              return null;
          }
      }
      
    • 实现类也是泛型类,实现类和接口类的泛型要一致,同样也可以进行扩充

      public class son<T> implements Father<T>{
          @Override
          public T getKey() {
              return null;
          }
      }
      
      public class son<T,K> implements Father<T>{
          private T Key;
          private K Value;
      
          @Override
          public T getKey() {
              return Key;
          }
      
          public K getValue(){
              return Value;
          }
      }
      

泛型方法

泛型类和泛型方法有什么区别呢?

  • 泛型类,是在实例化类的时候指明泛型的具体类型。
  • 泛型方法,是在调用方法的时候指明泛型的具体类型。

语法:

修饰符<T,K,...>返回值类型 方法名(形参列表){
  方法体
}
  • public 和返回值中间 非常重要,可以理解为生命此方法为泛型方法
  • 只有声明了的方法才是泛型方法,泛型类中的使用了泛型的成员方法并不是泛型方法
  • 表明该方法将使用泛型类型T,此时才可以在方法中使用泛型类型T
  • 与泛型类的定义一样,此处T可以写成任意标识,常见的如K,T,V,E等形式的参数常用于表示泛型。
public class Product<T> {
    Random ran = new Random();
    ArrayList<T> list = new ArrayList<>();
    public<E> E getProduct(ArrayList<E> list){ //如果不加<E> 其他两个E就会报错,因为这不是一个泛型方法,只能加了<E> 才能表明这是一个泛型方法,其余的两个E才不会报错
        return list.get(ran.nextInt(list.size()));
    }
    public void addList(T item){
        list.add(item);
    }

    public void getProduct(){
        System.out.println(list.get(ran.nextInt(list.size())));
    }
}
public class demo01{
    public static void main(String[] args) {
        Product<String> objectProduct = new Product<>();
        objectProduct.addList("苹果电脑");
        objectProduct.addList("华为手机");
        objectProduct.addList("扫地机器人");
        objectProduct.getProduct();
        ArrayList<Integer> list = new ArrayList<>();
        list.add(100);
        list.add(200);
        System.out.println(objectProduct.getProduct(list));
    }
}

类型通配符

什么是类型通配符?

  • 类型通配符一般是使用“?”代替具体的类型实参。
  • 所以,类型通配符是类型实参,而不是类型形参。

多态作为面向对象的三大特征之一,是不是在泛型上也可以使用多态的做法呢??(错的)

public class getBox {
    public static void main(String[] args) {
        Box<Integer> box1 = new Box<>();
        box1.setNum(200);
        showBox(box1);  //报错
        
        Box<Number> box2 = new Box<>();
        box2.setNum(100.0);
        showBox(box2);
    }

    public static  void showBox(Box<Number> box){
        box.getNum();
    }
}

通过上面两个例子可以发现,Integer 是Number 的 子类,但是这样写的话,会报错的,也就是泛型不支持多态的写法。

那既然上面这种办法不行,那按照我们之前学习java中学过的重载思想去做这题,我们写好几个相同的方法,只要里面的参数不一样不就行了吗,我把泛型改改是不是就能解决这种问题呢???????(错的)

public class getBox {
    public static void showBox(Box<Integer> box){ //报错
        box.getNum();
    }

    public static  void showBox(Box<Number> box){ //报错
        box.getNum();
    }
}

咱们之前学过方法的重载,认为只要参数不一样,相同的方法名他就是可以同时存在的,但放到泛型上面,用类似的方法去写,这是不可以的,虽然他们是泛型类型不同,但他们本质上都是相同类型。上面两个例子的本质上都是Box类型,因此他们不能够重载。

以上两种方法 多态和重载 都行不通,这样就大大的限制了泛型的灵活性,那么这个时候 类型通配符 就完美的解决了上面的两个问题。

我们只需要改成 下面这个样子就好

public class getBox {
    public static void main(String[] args) {
        Box<Integer> box1 = new Box<>();
        box1.setNum(200);
        showBox(box1);

        Box<Number> box2 = new Box<>();
        box2.setNum(100.0);
        showBox(box2);
    }

    public static  void showBox(Box<?> box){
        box.getNum();
    }
}

这个样子程序是能够完美运行的,通配符解决了刚刚的两个问题

为什么super叫下限?因为要从指定的参数往上看,extends为何叫上限,因为要从参数往下看。

类型通配符上限

  • 语法
/接口<? extends 实参类型>

要求该泛型类型,只能是实参类型,或实参类型的子类类型

public class getBox {
    public static void main(String[] args) {
        Box<Integer> box1 = new Box<>();
        box1.setNum(200);
        showBox(box1);

        Box<Number> box2 = new Box<>();
        box2.setNum(100.0);
        showBox(box2);
    }

    public static  void showBox(Box<? extends Number> box){
        box.getNum();
    }
}

类型通配符下限

  • 语法

    /接口<? super 实参类型>
        要求该泛型的类型,只能是实参类型,或实参类型的父类类型。
    

要求该泛型的类型,只能是实参类型,或实参类型的父类类型。

public class getBox {
    public static void main(String[] args) {
        Box<Integer> box1 = new Box<>();
        box1.setNum(200);
        showBox(box1);

        Box<Number> box2 = new Box<>();
        box2.setNum(100.0);
        showBox(box2);
    }

    public static  void showBox(Box<? super Integer> box){
        box.getNum();
    }
}

类型擦除

概念

泛型是Java1.5版本才引进的概念,在这之前是没有泛型的,但是,泛型代码能够很好的和之前版本的代码兼容,那是因为,泛型信息只存在代码编译阶段,再进去JVM之前,与泛型相关的信息会被擦除掉,我们称之为–类型擦除。

下面就是一个简单的类型擦除

public class demo01 {
    public static void main(String[] args) {
        ArrayList<String> list1 = new ArrayList<>();
        ArrayList<Integer> list2 = new ArrayList<>();

        System.out.println(list1.getClass().getSimpleName());//ArrayList
        System.out.println(list2.getClass().getSimpleName());//ArrayList

    }
}

无限制类型擦除

public class demo01 {
    public static void main(String[] args) {
        Erasure<Integer> erasure = new Erasure<>();
        Class<? extends Erasure> aClass = erasure.getClass();
        Field[] declaredFields = aClass.getDeclaredFields();
        for(Field i : declaredFields){
            System.out.println(i.getName() + ":" + i.getType().getSimpleName());
        }

        Method[] declaredMethods = aClass.getDeclaredMethods();
        for (Method j : declaredMethods){
            System.out.println(j.getName() + ":" + j.getReturnType().getSimpleName());
        }


    }
}

class Erasure<T>{
    public T key;

    public T getKey() {
        return key;
    }

    public void setKey(T key) {
        this.key = key;
    }
}

//key:Object
//getKey:Object
//setKey:void

有限制的类型擦除

public class demo01 {
    public static void main(String[] args) {
        Erasure<Integer> erasure = new Erasure<>();
        Class<? extends Erasure> aClass = erasure.getClass();
        Field[] declaredFields = aClass.getDeclaredFields();
        for(Field i : declaredFields){
            System.out.println(i.getName() + ":" + i.getType().getSimpleName());
        }

        Method[] declaredMethods = aClass.getDeclaredMethods();
        for (Method j : declaredMethods){
            System.out.println(j.getName() + ":" + j.getReturnType().getSimpleName());
        }


    }
}

class Erasure<T extends Number>{
    public T key;

    public T getKey() {
        return key;
    }

    public void setKey(T key) {
        this.key = key;
    }
}

//key:Number
//getKey:Number
//setKey:void

类型擦除,他只会往高处擦除,他不会往低处擦除,如果是用了super,他照样会转换成Obejct

方法的类型擦除

public class demo01 {
    public static void main(String[] args) {
        Erasure<Integer> erasure = new Erasure<>();
        Class<? extends Erasure> aClass = erasure.getClass();

        Method[] declaredMethods = aClass.getDeclaredMethods();
        for (Method j : declaredMethods){
            System.out.println(j.getName() + ":" + j.getReturnType().getSimpleName());
        }


    }
}

class Erasure<T extends Number>{

    public <T extends List> T show(T t){
        return t;
    }
}

 //show:List

桥接方法

不好描述,直接上代码示例

public interface Info<T>{
    T info(T var0;
}

public class InfoImpl implements Info<Integer>{
    @Overrdie
    public Integer info(Integer var){
        return var;
    }
}

过了编译以后

public interface Info{
    Object info(Object var);
}

public class InfoImpl implements Info{
    public Integer info(Integer var){
        return var;
    }
    
    //桥接方法,保持接口和类的实现关系
    @Override
    public Object info(Object var){
        retturn info((Integer)var);
    }
}

咱们看一个案例

public interface Box<T> {
    T info(T var);
}

public class Tom implements Box<Integer>{
    @Override
    public Integer info(Integer var) {
        return null;
    }
}

public class demo01 {
    public static void main(String[] args) {
        Tom tom = new Tom();
        Class<? extends Tom> aClass = tom.getClass();
        Method[] declaredMethods = aClass.getDeclaredMethods();
        for (Method j : declaredMethods){
            System.out.println(j.getName() + ":" + j.getReturnType().getSimpleName());
        }
    }
}

//info:Integer
//info:Object

泛型与数组

  • 可以声明带泛型的数组引用,但是不能直接创建带泛型的数组对象

  • 可以通过java.lang.reflect.Array的newInstance<Class,int> 创建T[] 数组

  • 针对第一句话:可以声明带泛型的数组引用,但是不能直接创建带泛型的数组对象

代码如下

public class demo01 {
    public static void main(String[] args) {
        ArrayList<String>[] arrList = new ArrayList<String>[6]; //报错的
    }
}

因为一个数组对象的类型是一直保存下去的,然而泛型是在编译的时候消除掉,他两个相互矛盾,所以就会报错。

那我们怎么去使用泛型数组呢,在使用过程中,我们可能会遇到下面的情况。下面情况就是抛出异常的。

Exception in thread “main” java.lang.ClassCastException: java.lang.Integer cannot be cast to java.lang.String

public class demo01 {
    public static void main(String[] args) {
        ArrayList[] list = new ArrayList[5];
        ArrayList<String>[] arrList = list;

        ArrayList<Integer> list1 = new ArrayList<>();
        list1.add(100);
        list1.add(200);

        list[0] = list1;
        String s = arrList[0].get(0);//在这我们就会发现,他怎么使用String类型接收的,那是因为我们是通过arrList这个引用去拿的,arrList引用是String类型的,所以才会产生这种情况,然而这种情况不是我们想要的结果。此时,在编译阶段,是不会报错的,只有运行后,才会发现错误
        System.out.println(s);//在这程序就会报错
    }
}

//代码就会报错
//Exception in thread "main" java.lang.ClassCastException: java.lang.Integer cannot //be cast to java.lang.String
//	at demo05.demo01.main(demo01.java:15)

那我们怎么去解决这个问题呢,我们就不要把ArrayList数组的原生引用给暴露出来,因为这样是无法展现泛型的特点(在编译阶段,将错误展现出来)改后的代码如下。

public class demo01 {
    public static void main(String[] args) {
        ArrayList<String>[] arrList = new ArrayList[5];

        ArrayList<Integer> list1 = new ArrayList<>();
        list1.add(100);
        list1.add(200);

        arrList[0] = list1;  //报错 因为ArrayList数组是String类型,然而你传入了一个Integer类型的集合,在编译阶段就会检查出来,大大的避免了程序的出错。
        String s = arrList[0].get(0);
    }
}

因为ArrayList数组是String类型,然而你传入了一个Integer类型的集合,在编译阶段就会检查出来,大大的避免了程序的出错

  • 可以通过java.lang.reflect.Array的newInstance<Class,int> 创建T[] 数组
public class Fruit<T> {
    private T[] array;

    public Fruit(Class<T> clz, int length){
        array = (T[]) Array.newInstance(clz,length);
    }

    public void put(int index,T item){
        array[index] = item;
    }

    public T get(int index){
        return array[index];
    }

    public T[] getArray(){
        return array;
    }
}

public class demo01 {
    public static void main(String[] args) {
        Fruit<String> fruit =new Fruit<>(String.class,3);
        fruit.put(0,"苹果");
        fruit.put(1,"西瓜");
        fruit.put(2,"香蕉");
        System.out.println(Arrays.toString(fruit.getArray()));
    }
}

//[苹果, 西瓜, 香蕉]
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

GaoJa

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值