Java 泛型(下)

简介: Java 泛型

Java 泛型(上):https://developer.aliyun.com/article/1491136


11.3 类型通配符


当我们声明一个变量/形参时,这个变量/形参的类型是一个泛型类或泛型接口,例如:Comparator类型,但是我们仍然无法确定这个泛型类或泛型接口的类型变量<T>的具体类型,此时我们考虑使用类型通配符。

public static void forList1(List<?> list){
    for (Object o : list) {
        System.out.println(o);
    }
}
//表示此方法可以接受一个泛型是<Animal>或者List<Animal的子类型>的List集合
public static void forList2(List<? extends Animal> list){
    for (Animal animal : list) {
        System.out.println(animal);
    }
}
public static <T> void forList3(List<? extends T> list){
    for (T o : list) {
        System.out.println(o);
    }
}
public static <T> void forList3_1(List<? extends T> list,T t){
    list.add(t);//错误,无法存入数据,因为传入的List集合泛型可以是T及子类类型,如果传入的List泛型是T的子类,那么T是放不进去这个集合的
    for (T o : list) {
        System.out.println(o);
    }
    return list.get(0);//可以取出T型元素
}
//表示此方法可以接受一个泛型是<Animal>或者List<Animal的父类型>的List集合
public static void forList4(List<? super Animal> list){
    for (Object o : list) {
        System.out.println(o);
    }
}
public static<T> void forList5(List<? super T> list){
    for (Object o : list) {
        System.out.println(o);
    }
}
public static <T> T forList5_1(List<? super T> list,T t){
    list.add(t);//可以存入数据,因为传入的List集合泛型可以是T及父类类型,如果传入的List泛型是T的父类,T也可以放入这个集合中
    for (T o : list) {
        System.out.println(o);
    }
    //return list.get(0);//错误,返回值类型为T,传入的List泛型可能为T的父类型,也就可能是其中元素为T的父类的List,所以取出来元素不一定是T型
    return null;
}

例如:

这个学生类是一个参数化的泛型类,代码如下(详细请看$11.2.1中的示例说明):

public class Student<T>{
  private String name;
  private T score;
  
  public Student() {
    super();
  }
  public Student(String name, T score) {
    super();
    this.name = name;
    this.score = score;
  }
  public String getName() {
    return name;
  }
  public void setName(String name) {
    this.name = name;
  }
  public T getScore() {
    return score;
  }
  public void setScore(T score) {
    this.score = score;
  }
  @Override
  public String toString() {
    return "姓名:" + name + ", 成绩:" + score;
  }
}


11.4.1 <?>任意类型


例如:我们要声明一个学生管理类,这个管理类要包含一个方法,可以遍历学生数组。

学生管理类:

class StudentService {
  public static void print(Student<?>[] arr) {
    for (int i = 0; i < arr.length; i++) {
      System.out.println(arr[i]);
    }
  }
}

测试类

public class TestGeneric {
  public static void main(String[] args) {
    // 语文老师使用时:
    Student<String> stu1 = new Student<String>("张三", "良好");
    // 数学老师使用时:
    // Student<double> stu2 = new Student<double>("张三", 90.5);//错误,必须是引用数据类型
    Student<Double> stu2 = new Student<Double>("张三", 90.5);
    // 英语老师使用时:
    Student<Character> stu3 = new Student<Character>("张三", 'C');
    Student<?>[] arr = new Student[3];
    arr[0] = stu1;
    arr[1] = stu2;
    arr[2] = stu3;
    StudentService.print(arr);
  }
}


11.4.2 <? extends 上限>


用于设定通配符上限

例如:我们要声明一个学生管理类,这个管理类要包含一个方法,找出学生数组中成绩最高的学生对象。

要求学生的成绩的类型必须可比较大小,实现Comparable接口。

学生管理类:

class StudentService {
    //分数score的类型必须是实现了Comparable接口的
  public static Student<? extends Comparable> max(Student<? extends Comparable>[] arr){
    Student<? extends Comparable> max = arr[0];
    for (int i = 0; i < arr.length; i++) {
      if(arr[i].getScore().compareTo(max.getScore())>0){
        max = arr[i];
      }
    }
    return max;
  }
}

测试类

public class TestGeneric {
  public static void main(String[] args) {
    Student<? extends Double>[] arr = new Student[3];
    arr[0] = new Student<Double>("张三", 90.5);
    arr[1] = new Student<Double>("李四", 80.5);
    arr[2] = new Student<Double>("王五", 94.5);
    
    Student<? extends Comparable> max = StudentService.max(arr);
    System.out.println(max);
  }
}


11.4.3 <? super 下限>


用于设定通配符下限

现在要声明一个数组工具类,包含可以给任意对象数组进行从小到大排序,只要你指定定制比较器对象,而且这个定制比较器对象可以是当前数组元素类型自己或其父类的定制比较器对象

数组工具类:

class MyArrays{
  public static <T> void sort(T[] arr, Comparator<? super T> c){
    for (int i = 1; i < arr.length; i++) {
      for (int j = 0; j < arr.length-i; j++) {
        if(c.compare(arr[j], arr[j+1])>0){
          T temp = arr[j];
          arr[j] = arr[j+1];
          arr[j+1] = temp;
        }
      }
    }
  }
}

例如:有如下JavaBean

class Person{
  private String name;
  private int age;
  public Person(String name, int age) {
    super();
    this.name = name;
    this.age = age;
  }
  public Person() {
    super();
  }
  public String getName() {
    return name;
  }
  public void setName(String name) {
    this.name = name;
  }
  public int getAge() {
    return age;
  }
  public void setAge(int age) {
    this.age = age;
  }
  @Override
  public String toString() {
    return "name=" + name + ", age=" + age;
  }
}
class Student extends Person{
  private int score;
  public Student(String name, int age, int score) {
    super(name, age);
    this.score = score;
  }
  public Student() {
    super();
  }
  public int getScore() {
    return score;
  }
  public void setScore(int score) {
    this.score = score;
  }
  @Override
  public String toString() {
    return super.toString() + ",score=" + score;
  }
  
}

测试类

public class TestGeneric {
  public static void main(String[] args) {
    Student[] all = new Student[3];
    all[0] = new Student("张三", 23, 89);
    all[1] = new Student("李四", 22, 99);
    all[2] = new Student("王五", 25, 67);
    
    MyArrays.sort(all, new Comparator<Person>() {
      @Override
      public int compare(Person o1, Person o2) {
        return o1.getAge() - o2.getAge();
      }
    });
    
    System.out.println(Arrays.toString(all));
    
    MyArrays.sort(all, new Comparator<Student>() {
      @Override
      public int compare(Student o1, Student o2) {
        return o1.getScore() - o2.getScore();
      }
    });
    System.out.println(Arrays.toString(all));
  }
}


练习:

在数组工具类中声明如下泛型方法:

(1)可以在任意类型的对象数组中,查找某个元素的下标,按照顺序查找,如果有重复的,就返回第一个找到的,如果没有返回-1

(2)可以在任意类型的对象数组中,查找最大值,要求元素必须实现Comparable接口

(3)可以在任意类型的对象数组中,查找最大值,按照指定定制比较器来比较元素大小

(4)可以给任意对象数组进行从小到大排序,要求数组元素类型必须实现Comparable接口

(5)可以给任意对象数组进行从小到大排序,只要你指定定制比较器对象,不要求数组元素实现Comparable接口

(6)可以将任意对象数组的元素拼接为一个字符串返回

public class MyArrays {
  //可以在任意类型的对象数组中,查找某个元素的下标,按照顺序查找,如果有重复的,就返回第一个找到的,如果没有返回-1
  public static <T> int find(T[] arr, T value) {
    for (int i = 0; i < arr.length; i++) {
      if(arr[i].equals(value)) {//使用==比较太严格,使用equals方法,因为任意对象都有equals方法
        return i;
      }
    }
    return -1;
  }
  
  //可以在任意类型的对象数组中,查找最大值,要求元素必须实现Comparable接口
  public static <T extends Comparable<? super T>> T max(T[] arr) {
    T max = arr[0];
    for (int i = 0; i < arr.length; i++) {
      if(max.compareTo(arr[i])<0) {//if(max < arr[i]) {
        max = arr[i];
      }
    }
    return max;
  }
  
  //可以在任意类型的对象数组中,查找最大值,按照指定定制比较器来比较元素大小
  public static <T> T max(T[] arr, Comparator<? super T> c) {
    T max = arr[0];
    for (int i = 0; i < arr.length; i++) {
      if(c.compare(max, arr[i])<0) {//if(max < arr[i]) {
        max = arr[i];
      }
    }
    return max;
  }
  
  //可以给任意对象数组进行从小到大排序,要求数组元素类型必须实现Comparable接口
  public static <T extends Comparable<? super T>> void sort(T[] arr) {
    for (int i = 0; i < arr.length-1; i++) {
      int minIndex = i;
      for (int j = i+1; j < arr.length; j++) {
        if(arr[minIndex].compareTo(arr[j])>0) {
          minIndex = j;
        }
      }
      if(minIndex!=i) {
        T temp = arr[minIndex];
        arr[minIndex] = arr[i];
        arr[i] = temp;
      }
    }
  }
  
  //可以给任意对象数组进行从小到大排序,只要你指定定制比较器对象,不要求数组元素实现Comparable接口
  public static <T> void sort(T[] arr, Comparator<? super T> c) {
    for (int i = 0; i < arr.length-1; i++) {
      int minIndex = i;
      for (int j = i+1; j < arr.length; j++) {
        if(c.compare(arr[minIndex],arr[j])>0) {
          minIndex = j;
        }
      }
      if(minIndex!=i) {
        T temp = arr[minIndex];
        arr[minIndex] = arr[i];
        arr[i] = temp;
      }
    }
  }
  
  //可以将任意对象数组的元素拼接为一个字符串返回
  public static <T> String toString(T[] arr) {
    String str = "[";
    for (int i = 0; i < arr.length; i++) {
      if(i==0) {
        str += arr[i];
      }else {
        str += "," + arr[i];
      }
    }
    str += "]";
    return str;
  }
}


通配符演示示例:

public class TestWildcardType {
    @Test
    public void test1() {
        List<String> list = new ArrayList<>();
        //使用只能处理泛型参数为String的List集合的方法
        handleStringList(list);
        List<?> list1 = new ArrayList<String>();
        List<?> list2 = new ArrayList<Integer>();
        List<?> list3 = new ArrayList<Object>();
        //可以处理任意泛型参数的List集合
        handleList(list1);
        handleList(list2);
        handleList(list3);
    }
    //定义一个方法,只能处理泛型参数为String的List集合
    public  void handleStringList(List<String> list) {
        for (int i = 0; i < list.size(); i++) {
            System.out.println(list.get(i));
        }
    }
    @Test
    public void test2() {
        //一、使用通配符?
        List<?> list;//泛型变量可以是任意类型
        list = new ArrayList<>();//泛型变量默认是Object类型的泛型
        list = new ArrayList<Object>();
        list = new ArrayList<String>();
        list = new ArrayList<Number>();
        list = new ArrayList<Integer>();
//        list.add(1);//compile error 编译器无法确定要add的真实类型,可能是<Character>或<Byte>,那么add(0.5)不可以,只有在方法被调用时才能确定。
        Object o = list.get(0);//无法确定元素类型,只能使用Object接收
        //二、设定通配符上限
        List<? extends Number> list1;//泛型变量必须是Number子类类型
        list1 = new ArrayList<>();//泛型变量默认是 Number类型
//        list1 = new ArrayList<Object>();//compile error
        list1 = new ArrayList<Number>();
        list1 = new ArrayList<Double>();
//        list1.add(1);//compile error 编译器无法确定要add的真实类型,可能是<Character>或<Byte>,那么add(0.5)不可以,只有在方法被调用时才能确定。
        Number number = list1.get(0);//可以使用Number接收,自动向上转型
        //所以设定了通配符上限,通常只能获取数据,即生产数据
        //三、设定通配符下限
        List<? super Number> list2;//泛型变量必须是Number父类类型
        list2 = new ArrayList<>();//泛型变量默认是 Number类型,
        list2 = new ArrayList<Object>();
        list2 = new ArrayList<Number>();
//        list2 = new ArrayList<Integer>();//compile error
        //
        list2.add(1.2);//编译器可以确定要add的真实类型一定是数值类型父类,那么add一个数值类型就不会有问题。
        Object object = list2.get(0);//返回值类型只能确定是Number的超类,编译器不确定具体类型,只能使用Object接收
        //所以设定通配符下限,通常用于添加数据,修改数据等即消费数据
    }
    //1.使用通配符,可以接收任意泛型的List集合
    public void handleList(List<?> list){
        for (int i = 0; i < list.size(); i++) {
            System.out.println(list.get(0));
        }
    }
    //2.设定通配符上限
    //定义一个方法,只能处理装有数值类型元素的List集合
    public void handleNumberList(List<? extends Number> list){
        double sum=0;
        for (Number number : list) {
            sum += number.doubleValue();
        }
        System.out.println(sum);
    }
    //3.设定通配符下限
    //定一个方法,需要一个比较器,只要能处理T类型数据的比较器就可以
    //比如要比较T[]的数组元素大小,那么就需要一个可以比较T类型元素或能比较T的父类型元素(一定能比较T)的比较器
    public <T> void  handleComparator(T[] arr,Comparator<? super T> c){
    }
    //3.1设定通配符下限
    //定义一个方法,可以向任意List集合中(泛型类型下限为T型的集合),添加T型元素(泛型类型下限为T型的集合一定可以添加T型元素)
    public static <T> void fill(List<? super T> list,T obj){
        for (int i = 0; i < list.size(); i++) {
            list.add(obj);
        }
    }
}
();
        list2 = new ArrayList<Number>();
//        list2 = new ArrayList<Integer>();//compile error
        //
        list2.add(1.2);//编译器可以确定要add的真实类型一定是数值类型父类,那么add一个数值类型就不会有问题。
        Object object = list2.get(0);//返回值类型只能确定是Number的超类,编译器不确定具体类型,只能使用Object接收
        //所以设定通配符下限,通常用于添加数据,修改数据等即消费数据
    }
    //1.使用通配符,可以接收任意泛型的List集合
    public void handleList(List<?> list){
        for (int i = 0; i < list.size(); i++) {
            System.out.println(list.get(0));
        }
    }
    //2.设定通配符上限
    //定义一个方法,只能处理装有数值类型元素的List集合
    public void handleNumberList(List<? extends Number> list){
        double sum=0;
        for (Number number : list) {
            sum += number.doubleValue();
        }
        System.out.println(sum);
    }
    //3.设定通配符下限
    //定一个方法,需要一个比较器,只要能处理T类型数据的比较器就可以
    //比如要比较T[]的数组元素大小,那么就需要一个可以比较T类型元素或能比较T的父类型元素(一定能比较T)的比较器
    public <T> void  handleComparator(T[] arr,Comparator<? super T> c){
    }
    //3.1设定通配符下限
    //定义一个方法,可以向任意List集合中(泛型类型下限为T型的集合),添加T型元素(泛型类型下限为T型的集合一定可以添加T型元素)
    public static <T> void fill(List<? super T> list,T obj){
        for (int i = 0; i < list.size(); i++) {
            list.add(obj);
        }
    }
}


相关文章
|
1月前
|
安全 Java 编译器
揭秘JAVA深渊:那些让你头大的最晦涩知识点,从泛型迷思到并发陷阱,你敢挑战吗?
【8月更文挑战第22天】Java中的难点常隐藏在其高级特性中,如泛型与类型擦除、并发编程中的内存可见性及指令重排,以及反射与动态代理等。这些特性虽强大却也晦涩,要求开发者深入理解JVM运作机制及计算机底层细节。例如,泛型在编译时检查类型以增强安全性,但在运行时因类型擦除而丢失类型信息,可能导致类型安全问题。并发编程中,内存可见性和指令重排对同步机制提出更高要求,不当处理会导致数据不一致。反射与动态代理虽提供运行时行为定制能力,但也增加了复杂度和性能开销。掌握这些知识需深厚的技术底蕴和实践经验。
50 2
|
11天前
|
Java 编译器 容器
Java——包装类和泛型
包装类是Java中一种特殊类,用于将基本数据类型(如 `int`、`double`、`char` 等)封装成对象。这样做可以利用对象的特性和方法。Java 提供了八种基本数据类型的包装类:`Integer` (`int`)、`Double` (`double`)、`Byte` (`byte`)、`Short` (`short`)、`Long` (`long`)、`Float` (`float`)、`Character` (`char`) 和 `Boolean` (`boolean`)。包装类可以通过 `valueOf()` 方法或自动装箱/拆箱机制创建。
23 9
Java——包装类和泛型
|
14天前
|
安全 Java API
【Java面试题汇总】Java基础篇——String+集合+泛型+IO+异常+反射(2023版)
String常量池、String、StringBuffer、Stringbuilder有什么区别、List与Set的区别、ArrayList和LinkedList的区别、HashMap底层原理、ConcurrentHashMap、HashMap和Hashtable的区别、泛型擦除、ABA问题、IO多路复用、BIO、NIO、O、异常处理机制、反射
【Java面试题汇总】Java基础篇——String+集合+泛型+IO+异常+反射(2023版)
|
11天前
|
存储 安全 搜索推荐
Java中的泛型
【9月更文挑战第15天】在 Java 中,泛型是一种编译时类型检查机制,通过使用类型参数提升代码的安全性和重用性。其主要作用包括类型安全,避免运行时类型转换错误,以及代码重用,允许编写通用逻辑。泛型通过尖括号 `&lt;&gt;` 定义类型参数,并支持上界和下界限定,以及无界和有界通配符。使用泛型需注意类型擦除、无法创建泛型数组及基本数据类型的限制。泛型显著提高了代码的安全性和灵活性。
|
1月前
|
安全 Java Go
Java&Go泛型对比
总的来说,Java和Go在泛型的实现和使用上各有特点,Java的泛型更注重于类型安全和兼容性,而Go的泛型在保持类型安全的同时,提供了更灵活的类型参数和类型集的概念,同时避免了运行时的性能开销。开发者在使用时可以根据自己的需求和语言特性来选择使用哪种语言的泛型特性。
37 7
|
1月前
|
存储 算法 Java
14 Java集合(集合框架+泛型+ArrayList类+LinkedList类+Vector类+HashSet类等)
14 Java集合(集合框架+泛型+ArrayList类+LinkedList类+Vector类+HashSet类等)
41 2
14 Java集合(集合框架+泛型+ArrayList类+LinkedList类+Vector类+HashSet类等)
|
1月前
|
存储 安全 Java
如何理解java的泛型这个概念
理解java的泛型这个概念
|
1月前
|
存储 缓存 Java
|
1月前
|
安全 Java
【Java 第六篇章】泛型
Java泛型是自J2 SE 1.5起的新特性,允许类型参数化,提高代码复用性与安全性。通过定义泛型类、接口或方法,可在编译时检查类型安全,避免运行时类型转换异常。泛型使用尖括号`&lt;&gt;`定义,如`class MyClass&lt;T&gt;`。泛型方法的格式为`public &lt;T&gt; void methodName()`。通配符如`?`用于不确定的具体类型。示例代码展示了泛型类、接口及方法的基本用法。
11 0
|
1月前
|
Java
【Java基础面试四十五】、 介绍一下泛型擦除
这篇文章解释了Java泛型的概念,它解决了集合类型安全问题,允许在创建集合时指定元素类型,避免了类型转换的复杂性和潜在的异常。