第12章 泛型

学习目标

  • 能够使用泛型定义类、接口、方法
  • 能够理解泛型上限
  • 能够阐述泛型通配符的作用
  • 能够识别通配符的上下限

第十二章 泛型

12.1 泛型的概念

12.1.1 泛型的引入

例如:生产瓶子的厂家,一开始并不知道我们将来会用瓶子装什么,我们什么都可以装,但是有的时候,我们在使用时,想要限定某个瓶子只能用来装什么,这样我们不会装错,而用的时候也可以放心的使用,无需再三思量。我们生活中是在使用这个瓶子时在瓶子上“贴标签”,这样就轻松解决了问题。

尚硅谷_高佳志_JavaSE_第12章 泛型 - 图1

还有,在Java中我们在声明方法时,当在完成方法功能时如果有未知的数据需要参与,这些未知的数据需要在调用方法时才能确定,那么我们把这样的数据通过形参表示。那么在方法体中,用这个形参名来代表那个未知的数据,而调用者在调用时,对应的传入值就可以了。

尚硅谷_高佳志_JavaSE_第12章 泛型 - 图2

受以上两点启发,JDK1.5设计了泛型的概念。泛型即为“类型参数”,这个类型参数在声明它的类、接口或方法中,代表未知的通用的类型。例如:

java.lang.Comparable接口和java.util.Comparator接口,是用于对象比较大小的规范接口,这两个接口只是限定了当一个对象大于另一个对象时返回正整数,小于返回负整数,等于返回0。但是并不确定是什么类型的对象比较大小,之前的时候只能用Object类型表示,使用时既麻烦又不安全,因此JDK1.5就给它们增加了泛型。

  1. public interface Comparable<T>{
  2. int compareTo(T o) ;
  3. }
  1. public interface Comparator<T>{
  2. int compare(T o1, T o2) ;
  3. }

其中就是类型参数,即泛型。

12.1.2 泛型的好处

示例代码:

JavaBean:圆类型

  1. class Circle{
  2. private double radius;
  3. public Circle(double radius) {
  4. super();
  5. this.radius = radius;
  6. }
  7. public double getRadius() {
  8. return radius;
  9. }
  10. public void setRadius(double radius) {
  11. this.radius = radius;
  12. }
  13. @Override
  14. public String toString() {
  15. return "Circle [radius=" + radius + "]";
  16. }
  17. }

比较器

  1. import java.util.Comparator;
  2. public class CircleComparator implements Comparator{
  3. @Override
  4. public int compare(Object o1, Object o2) {
  5. //强制类型转换
  6. Circle c1 = (Circle) o1;
  7. Circle c2 = (Circle) o2;
  8. return Double.compare(c1.getRadius(), c2.getRadius());
  9. }
  10. }

测试类

  1. public class TestGeneric {
  2. public static void main(String[] args) {
  3. CircleComparator com = new CircleComparator();
  4. System.out.println(com.compare(new Circle(1), new Circle(2)));
  5. System.out.println(com.compare("圆1", "圆2"));//运行时异常:ClassCastException
  6. }
  7. }

那么我们在使用如上面这样的接口时,如果没有泛型或不指定泛型,很麻烦,而且有安全隐患。

因为在设计(编译)Comparator接口时,不知道它会用于哪种类型的对象比较,因此只能将compare方法的形参设计为Object类型,而实际在compare方法中需要向下转型为Circle,才能调用Circle类的getRadius()获取半径值进行比较。

使用泛型:

比较器:

  1. class CircleComparator implements Comparator<Circle>{
  2. @Override
  3. public int compare(Circle o1, Circle o2) {
  4. //不再需要强制类型转换,代码更简洁
  5. return Double.compare(o1.getRadius(), o2.getRadius());
  6. }
  7. }

测试类

  1. import java.util.Comparator;
  2. public class TestGeneric {
  3. public static void main(String[] args) {
  4. CircleComparator com = new CircleComparator();
  5. System.out.println(com.compare(new Circle(1), new Circle(2)));
  6. // System.out.println(com.compare("圆1", "圆2"));//编译错误,因为"圆1", "圆2"不是Circle类型,是String类型,编译器提前报错,而不是冒着风险在运行时再报错
  7. }
  8. }

如果有了泛型并使用泛型,那么既能保证安全,又能简化代码。

因为把不安全的因素在编译期间就排除了;既然通过了编译,那么类型一定是符合要求的,就避免了类型转换。

12.1.3 泛型的相关名词

<类型>这种语法形式就叫泛型。

其中:

  • 是类型变量(Type Variables),而是代表未知的数据类型,我们可以指定为,,等,那么<类型>的形式我们成为类型参数;

    • 类比方法的参数的概念,我们可以把,称为类型形参,将称为类型实参,有助于我们理解泛型;
  • Comparator这种就称为参数化类型(Parameterized Types)。

自从有了泛型之后,Java的数据类型就更丰富了:

尚硅谷_高佳志_JavaSE_第12章 泛型 - 图3

Class:Class 类的实例表示正在运行的 Java 应用程序中的类和接口。枚举是一种类,注释是一种接口。每个数组属于被映射为 Class 对象的一个类,所有具有相同元素类型和维数的数组都共享该 Class 对象。基本的 Java 类型(booleanbytecharshortintlongfloatdouble)和关键字 void 也表示为 Class 对象。

  • GenericArrayType:泛化的数组类型,即T[]
  • ParameterizedType:参数化类型,例如:Comparator,Comparator
  • TypeVariable:类型变量,例如:Comparator中的T,Map中的K,V
  • WildcardType:通配符类型,例如:Comparator<?>等

12.1.4 在哪里可以声明类型变量

  • 声明类或接口时,在类名或接口名后面声明类型变量,我们把这样的类或接口称为泛型类或泛型接口
  1. 【修饰符】 class 类名<类型变量列表> extends 父类】 implements 父接口们】{
  2. }
  3. 【修饰符】 interface 接口名<类型变量列表> implements 父接口们】{
  4. }
  5. 例如:
  6. public class ArrayList<E>
  7. public interface Map<K,V>{
  8. ....
  9. }
  • 声明方法时,在【修饰符】与返回值类型之间声明类型变量,我们把声明(是声明不是单纯的使用)了类型变量的方法称为泛型方法
  1. 【修饰符】 <类型变量列表> 返回值类型 方法名(【形参列表】)【throws 异常列表】{
  2. //...
  3. }
  4. 例如:java.util.Arrays类中的
  5. public static <T> List<T> asList(T... a){
  6. ....
  7. }

12.2 参数类型:泛型类与泛型接口

当我们在声明类或接口时,类或接口中定义某个成员时,该成员有些类型是不确定的,而这个类型需要在使用这个类或接口时才可以确定,那么我们可以使用泛型。

12.2.1 声明泛型类与泛型接口

语法格式:

  1. 【修饰符】 class 类名<类型变量列表> extends 父类】 implements 父接口们】{
  2. }
  3. 【修饰符】 interface 接口名<类型变量列表> implements 父接口们】{
  4. }

注意:

  • <类型变量列表>:可以是一个或多个类型变量,一般都是使用单个的大写字母表示。例如:、等。
  • <类型变量列表>中的类型变量不能用于静态成员上。

什么时候使用泛型类或泛型接口呢?

  • 当某个类的非静态实例变量的类型不确定,需要在创建对象或子类继承时才能确定
  • 当某个(些)类的非静态方法的形参类型不确定,需要在创建对象或子类继承时才能确定

示例代码:

例如:我们要声明一个学生类,该学生包含姓名、成绩,而此时学生的成绩类型不确定,为什么呢,因为,语文老师希望成绩是“优秀”、“良好”、“及格”、“不及格”,数学老师希望成绩是89.5, 65.0,英语老师希望成绩是’A’,’B’,’C’,’D’,’E’。那么我们在设计这个学生类时,就可以使用泛型。

  1. public class Student<T>{
  2. private String name;
  3. private T score;
  4. public Student() {
  5. super();
  6. }
  7. public Student(String name, T score) {
  8. super();
  9. this.name = name;
  10. this.score = score;
  11. }
  12. public String getName() {
  13. return name;
  14. }
  15. public void setName(String name) {
  16. this.name = name;
  17. }
  18. public T getScore() {
  19. return score;
  20. }
  21. public void setScore(T score) {
  22. this.score = score;
  23. }
  24. @Override
  25. public String toString() {
  26. return "姓名:" + name + ", 成绩:" + score;
  27. }
  28. }

12.2.2 使用泛型类与泛型接口

在使用这种参数化的类与接口时,我们需要指定泛型变量的实际类型参数:

(1)实际类型参数必须是引用数据类型,不能是基本数据类型

(2)在创建类的对象时指定类型变量对应的实际类型参数

  1. public class TestGeneric{
  2. public static void main(String[] args) {
  3. //语文老师使用时:
  4. Student<String> stu1 = new Student<String>("张三", "良好");
  5. //数学老师使用时:
  6. //Student<double> stu2 = new Student<double>("张三", 90.5);//错误,必须是引用数据类型
  7. Student<Double> stu2 = new Student<Double>("张三", 90.5);
  8. //英语老师使用时:
  9. Student<Character> stu3 = new Student<Character>("张三", 'C');
  10. //错误的指定
  11. //Student<Object> stu = new Student<String>();//错误的
  12. }
  13. }

JDK1.7支持简写形式:Student stu1 = new Student<>(“张三”, “良好”);

指定泛型实参时,必须左右两边一致,不存在多态现象

(3)在继承泛型类或实现泛型接口时,指定类型变量对应的实际类型参数

  1. class ChineseStudent extends Student<String>{
  2. public ChineseStudent() {
  3. super();
  4. }
  5. public ChineseStudent(String name, String score) {
  6. super(name, score);
  7. }
  8. }
  1. public class TestGeneric{
  2. public static void main(String[] args) {
  3. //语文老师使用时:
  4. ChineseStudent stu = new ChineseStudent("张三", "良好");
  5. }
  6. }
  1. class Circle implements Comparable<Circle>{
  2. private double radius;
  3. public Circle(double radius) {
  4. super();
  5. this.radius = radius;
  6. }
  7. public double getRadius() {
  8. return radius;
  9. }
  10. public void setRadius(double radius) {
  11. this.radius = radius;
  12. }
  13. @Override
  14. public String toString() {
  15. return "Circle [radius=" + radius + "]";
  16. }
  17. @Override
  18. public int compareTo(Circle c){
  19. return Double.compare(radius,c.radius);
  20. }
  21. }

12.2.3 类型变量的上限

当在声明类型变量时,如果不希望这个类型变量代表任意引用数据类型,而是某个系列的引用数据类型,那么可以设定类型变量的上限。

语法格式:

  1. <类型变量 extends 上限>

如果有多个上限

  1. <类型变量 extends 上限1 & 上限2>

如果多个上限中有类有接口,那么只能有一个类,而且必须写在最左边。接口的话,可以多个。

如果在声明<类型变量>时没有指定任何上限,默认上限是java.lang.Object。

例如:我们要声明一个两个数求和的工具类,要求两个加数必须是Number数字类型,并且实现Comparable接口。

  1. class SumTools<T extends Number & Comparable<T>>{
  2. private T a;
  3. private T b;
  4. public SumTools(T a, T b) {
  5. super();
  6. this.a = a;
  7. this.b = b;
  8. }
  9. @SuppressWarnings("unchecked")
  10. public T getSum(){
  11. if(a instanceof BigInteger){
  12. return (T) ((BigInteger) a).add((BigInteger)b);
  13. }else if(a instanceof BigDecimal){
  14. return (T) ((BigDecimal) a).add((BigDecimal)b);
  15. }else if(a instanceof Integer){
  16. return (T)(Integer.valueOf((Integer)a+(Integer)b));
  17. }else if(a instanceof Long){
  18. return (T)(Long.valueOf((Long)a+(Long)b));
  19. }else if(a instanceof Float){
  20. return (T)(Float.valueOf((Float)a+(Float)b));
  21. }else if(a instanceof Double){
  22. return (T)(Double.valueOf((Double)a+(Double)b));
  23. }
  24. throw new UnsupportedOperationException("不支持该操作");
  25. }
  26. }

测试类

  1. public static void main(String[] args) {
  2. SumTools<Integer> s = new SumTools<Integer>(1,2);
  3. Integer sum = s.getSum();
  4. System.out.println(sum);
  5. // SumTools<String> s = new SumTools<String>("1","2");//错误,因为String类型不是extends Number
  6. }

12.2.4 泛型擦除

当使用参数化类型的类或接口时,如果没有指定泛型,那么会怎么样呢?

会发生泛型擦除,自动按照最左边的第一个上限处理。如果没有指定上限,上限即为Object。

  1. public static void main(String[] args) {
  2. SumTools s = new SumTools(1,2);
  3. Number sum = s.getSum();
  4. System.out.println(sum);
  5. }
  1. import java.util.Comparator;
  2. public class CircleComparator implements Comparator{
  3. @Override
  4. public int compare(Object o1, Object o2) {
  5. //强制类型转换
  6. Circle c1 = (Circle) o1;
  7. Circle c2 = (Circle) o2;
  8. return Double.compare(c1.getRadius(), c2.getRadius());
  9. }
  10. }

12.2.5 练习

练习1

1、声明一个坐标类Coordinate,它有两个属性:x,y,都为T类型
2、在测试类中,创建两个不同的坐标类对象,
分别指定T类型为String和Double,并为x,y赋值,打印对象

  1. public class TestExer1 {
  2. public static void main(String[] args) {
  3. Coordinate<String> c1 = new Coordinate<>("北纬38.6", "东经36.8");
  4. System.out.println(c1);
  5. // Coordinate<Double> c2 = new Coordinate<>(38.6, 38);//自动装箱与拆箱只能与对应的类型 38是int,自动装为Integer
  6. Coordinate<Double> c2 = new Coordinate<>(38.6, 36.8);
  7. System.out.println(c2);
  8. }
  9. }
  10. class Coordinate<T>{
  11. private T x;
  12. private T y;
  13. public Coordinate(T x, T y) {
  14. super();
  15. this.x = x;
  16. this.y = y;
  17. }
  18. public Coordinate() {
  19. super();
  20. }
  21. public T getX() {
  22. return x;
  23. }
  24. public void setX(T x) {
  25. this.x = x;
  26. }
  27. public T getY() {
  28. return y;
  29. }
  30. public void setY(T y) {
  31. this.y = y;
  32. }
  33. @Override
  34. public String toString() {
  35. return "Coordinate [x=" + x + ", y=" + y + "]";
  36. }
  37. }

练习2

1、声明一个Person类,包含姓名和伴侣属性,其中姓名是String类型,而伴侣的类型不确定,
因为伴侣可以是Person,可以是Animal(例如:金刚),可以是Ghost鬼(例如:倩女幽魂),
可以是Demon妖(例如:白娘子),可以是Robot机器人(例如:剪刀手爱德华)。。。

2、在测试类中,创建Person对象,并为它指定伴侣,打印显示信息

  1. public class TestExer3 {
  2. @SuppressWarnings({ "rawtypes", "unchecked" })
  3. public static void main(String[] args) {
  4. Person<Demon> xu = new Person<Demon>("许仙",new Demon("白娘子"));
  5. System.out.println(xu);
  6. Person<Person> xie = new Person<Person>("谢学建",new Person("徐余龙"));
  7. Person fere = xie.getFere();
  8. fere.setFere(xie);
  9. System.out.println(xie);
  10. System.out.println(fere);
  11. }
  12. }
  13. class Demon{
  14. private String name;
  15. public Demon(String name) {
  16. super();
  17. this.name = name;
  18. }
  19. @Override
  20. public String toString() {
  21. return "Demon [name=" + name + "]";
  22. }
  23. }
  24. class Person<T>{
  25. private String name;
  26. private T fere;
  27. public Person(String name, T fere) {
  28. super();
  29. this.name = name;
  30. this.fere = fere;
  31. }
  32. public Person(String name) {
  33. super();
  34. this.name = name;
  35. }
  36. public Person() {
  37. super();
  38. }
  39. public String getName() {
  40. return name;
  41. }
  42. public void setName(String name) {
  43. this.name = name;
  44. }
  45. public T getFere() {
  46. return fere;
  47. }
  48. public void setFere(T fere) {
  49. this.fere = fere;
  50. }
  51. @SuppressWarnings("rawtypes")
  52. @Override
  53. public String toString() {
  54. if(fere instanceof Person){
  55. Person p = (Person) fere;
  56. return "Person [name=" + name + ", fere=" + p.getName() + "]";
  57. }
  58. return "Person [name=" + name + ", fere=" + fere + "]";
  59. }
  60. }

练习3

1、声明员工类型Employee,包含姓名(String),薪资(double),年龄(int)

2、员工类Employee实现java.lang.Comparable接口,指定T为Employee类型,重写抽象方法,按照薪资比较大小,薪资相同的按照姓名的自然顺序比较大小。

3、在测试类中创建Employee数组,然后调用Arrays.sort(Object[] arr)方法进行排序,遍历显示员工信息

4、再次调用Arrays.sort(Object[] arr,Comparator c)方法进行按照年龄排序,年龄相同的安装姓名自然顺序比较大小,遍历显示员工信息

  1. public class TestExer3 {
  2. @Test
  3. public void test01() {
  4. Employee[] arr = new Employee[3];
  5. arr[0] = new Employee("Irene", 18000, 18);
  6. arr[1] = new Employee("Jack", 14000, 28);
  7. arr[2] = new Employee("Alice", 14000, 24);
  8. Arrays.sort(arr);
  9. for (int i = 0; i < arr.length; i++) {
  10. System.out.println(arr[i]);
  11. }
  12. }
  13. @Test
  14. public void test02() {
  15. Employee[] arr = new Employee[3];
  16. arr[0] = new Employee("Irene", 18000, 18);
  17. arr[1] = new Employee("Jack", 14000, 28);
  18. arr[2] = new Employee("Alice", 14000, 24);
  19. //Arrays.sort(T[] arr,Comparator<T> c)
  20. Arrays.sort(arr, new Comparator<Employee>() {
  21. //按照年龄排序,年龄相同的安装姓名自然顺序比较大小
  22. @Override
  23. public int compare(Employee o1, Employee o2) {
  24. if(o1.getAge() != o2.getAge()) {
  25. return o1.getAge() - o2.getAge();
  26. }
  27. return o1.getName().compareTo(o2.getName());
  28. }
  29. });
  30. for (int i = 0; i < arr.length; i++) {
  31. System.out.println(arr[i]);
  32. }
  33. }
  34. }
  35. class Employee implements Comparable<Employee>{
  36. private String name;
  37. private double salary;
  38. private int age;
  39. public Employee(String name, double salary, int age) {
  40. super();
  41. this.name = name;
  42. this.salary = salary;
  43. this.age = age;
  44. }
  45. public Employee() {
  46. super();
  47. }
  48. public String getName() {
  49. return name;
  50. }
  51. public void setName(String name) {
  52. this.name = name;
  53. }
  54. public double getSalary() {
  55. return salary;
  56. }
  57. public void setSalary(double salary) {
  58. this.salary = salary;
  59. }
  60. public int getAge() {
  61. return age;
  62. }
  63. public void setAge(int age) {
  64. this.age = age;
  65. }
  66. @Override
  67. public String toString() {
  68. return "Employee [name=" + name + ", salary=" + salary + ", age=" + age + "]";
  69. }
  70. //重写抽象方法,按照薪资比较大小,薪资相同的按照姓名的自然顺序比较大小。
  71. @Override
  72. public int compareTo(Employee o) {
  73. if(this.salary != o.salary) {
  74. return Double.compare(this.salary, o.salary);
  75. }
  76. return this.name.compareTo(o.name);//name是String类型,有compareTo方法
  77. }
  78. }

12.3 泛型方法

前面介绍了在定义类、接口时可以声明<类型变量>,在该类的方法和属性定义、接口的方法定义中,这些<类型变量>可被当成普通类型来用。但是,在另外一些情况下,

(1)如果我们定义类、接口时没有使用<类型变量>,但是某个方法形参类型不确定时,可以单独这个方法定义<类型变量>;

(2)另外我们之前说类和接口上的类型形参是不能用于静态方法中,那么当某个静态方法的形参类型不确定时,可以单独定义<类型变量>。

那么,JDK1.5之后,还提供了泛型方法的支持。

语法格式:

  1. 【修饰符】 <类型变量列表> 返回值类型 方法名(【形参列表】)【throws 异常列表】{
  2. //...
  3. }
  • <类型变量列表>:可以是一个或多个类型变量,一般都是使用单个的大写字母表示。例如:、等。
  • <类型变量>同样也可以指定上限

示例代码:

我们编写一个数组工具类,包含可以给任意对象数组进行从小到大排序,要求数组元素类型必须实现Comparable接口

  1. public class MyArrays{
  2. public static <T extends Comparable<T>> void sort(T[] arr){
  3. for (int i = 1; i < arr.length; i++) {
  4. for (int j = 0; j < arr.length-i; j++) {
  5. if(arr[j].compareTo(arr[j+1])>0){
  6. T temp = arr[j];
  7. arr[j] = arr[j+1];
  8. arr[j+1] = temp;
  9. }
  10. }
  11. }
  12. }
  13. }

测试类

  1. public class TestGeneric{
  2. public static void main(String[] args) {
  3. int[] arr = {3,2,5,1,4};
  4. // MyArrays.sort(arr);//错误的,因为int[]不是对象数组
  5. String[] strings = {"hello","java","chai"};
  6. MyArrays.sort(strings);
  7. System.out.println(Arrays.toString(strings));
  8. Circle[] circles = {new Circle(2.0),new Circle(1.2),new Circle(3.0)};
  9. MyArrays.sort(circles);
  10. System.out.println(Arrays.toString(circles));
  11. }
  12. }

12.4 类型通配符

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

例如:

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

  1. public class Student<T>{
  2. private String name;
  3. private T score;
  4. public Student() {
  5. super();
  6. }
  7. public Student(String name, T score) {
  8. super();
  9. this.name = name;
  10. this.score = score;
  11. }
  12. public String getName() {
  13. return name;
  14. }
  15. public void setName(String name) {
  16. this.name = name;
  17. }
  18. public T getScore() {
  19. return score;
  20. }
  21. public void setScore(T score) {
  22. this.score = score;
  23. }
  24. @Override
  25. public String toString() {
  26. return "姓名:" + name + ", 成绩:" + score;
  27. }
  28. }

12.4.1 <?>任意类型

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

学生管理类:

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

测试类

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

12.4.2 <? extends 上限>

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

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

学生管理类:

  1. class StudentService {
  2. @SuppressWarnings({ "rawtypes", "unchecked" })
  3. public static Student<? extends Comparable> max(Student<? extends Comparable>[] arr){
  4. Student<? extends Comparable> max = arr[0];
  5. for (int i = 0; i < arr.length; i++) {
  6. if(arr[i].getScore().compareTo(max.getScore())>0){
  7. max = arr[i];
  8. }
  9. }
  10. return max;
  11. }
  12. }

测试类

  1. public class TestGeneric {
  2. @SuppressWarnings({ "rawtypes", "unchecked" })
  3. public static void main(String[] args) {
  4. Student<? extends Double>[] arr = new Student[3];
  5. arr[0] = new Student<Double>("张三", 90.5);
  6. arr[1] = new Student<Double>("李四", 80.5);
  7. arr[2] = new Student<Double>("王五", 94.5);
  8. Student<? extends Comparable> max = StudentService.max(arr);
  9. System.out.println(max);
  10. }
  11. }

12.4.3 <? super 下限>

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

数组工具类:

  1. class MyArrays{
  2. public static <T> void sort(T[] arr, Comparator<? super T> c){
  3. for (int i = 1; i < arr.length; i++) {
  4. for (int j = 0; j < arr.length-i; j++) {
  5. if(c.compare(arr[j], arr[j+1])>0){
  6. T temp = arr[j];
  7. arr[j] = arr[j+1];
  8. arr[j+1] = temp;
  9. }
  10. }
  11. }
  12. }
  13. }

例如:有如下JavaBean

  1. class Person{
  2. private String name;
  3. private int age;
  4. public Person(String name, int age) {
  5. super();
  6. this.name = name;
  7. this.age = age;
  8. }
  9. public Person() {
  10. super();
  11. }
  12. public String getName() {
  13. return name;
  14. }
  15. public void setName(String name) {
  16. this.name = name;
  17. }
  18. public int getAge() {
  19. return age;
  20. }
  21. public void setAge(int age) {
  22. this.age = age;
  23. }
  24. @Override
  25. public String toString() {
  26. return "name=" + name + ", age=" + age;
  27. }
  28. }
  29. class Student extends Person{
  30. private int score;
  31. public Student(String name, int age, int score) {
  32. super(name, age);
  33. this.score = score;
  34. }
  35. public Student() {
  36. super();
  37. }
  38. public int getScore() {
  39. return score;
  40. }
  41. public void setScore(int score) {
  42. this.score = score;
  43. }
  44. @Override
  45. public String toString() {
  46. return super.toString() + ",score=" + score;
  47. }
  48. }

测试类

  1. public class TestGeneric {
  2. public static void main(String[] args) {
  3. Student[] all = new Student[3];
  4. all[0] = new Student("张三", 23, 89);
  5. all[1] = new Student("李四", 22, 99);
  6. all[2] = new Student("王五", 25, 67);
  7. MyArrays.sort(all, new Comparator<Person>() {
  8. @Override
  9. public int compare(Person o1, Person o2) {
  10. return o1.getAge() - o2.getAge();
  11. }
  12. });
  13. System.out.println(Arrays.toString(all));
  14. MyArrays.sort(all, new Comparator<Student>() {
  15. @Override
  16. public int compare(Student o1, Student o2) {
  17. return o1.getScore() - o2.getScore();
  18. }
  19. });
  20. System.out.println(Arrays.toString(all));
  21. }
  22. }

12.4.4 使用类型通配符来指定类型参数的问题

  1. stu1.setScore(null);//除了null,无法设置为其他值
  2. Student<? extends Number> stu2 = new Student<>();
  3. stu2.setScore(null);//除了null,无法设置为其他值
  4. Student<? super Number> stu3 = new Student<>();
  5. stu3.setScore(56);//可以设置Number或其子类的对象
  6. }

}
class Student{
private String name;
private T score;

  1. public Student() {
  2. super();
  3. }
  4. public Student(String name, T score) {
  5. super();
  6. this.name = name;
  7. this.score = score;
  8. }
  9. public String getName() {
  10. return name;
  11. }
  12. public void setName(String name) {
  13. this.name = name;
  14. }
  15. public T getScore() {
  16. return score;
  17. }
  18. public void setScore(T score) {
  19. this.score = score;
  20. }
  21. @Override
  22. public String toString() {
  23. return "姓名:" + name + ", 成绩:" + score;
  24. }

}

  1. ## 12.5 练习
  2. 在数组工具类中声明如下泛型方法:
  3. 1)可以在任意类型的对象数组中,查找某个元素的下标,按照顺序查找,如果有重复的,就返回第一个找到的,如果没有返回-1
  4. 2)可以在任意类型的对象数组中,查找最大值,要求元素必须实现Comparable接口
  5. 3)可以在任意类型的对象数组中,查找最大值,按照指定定制比较器来比较元素大小
  6. 4)可以给任意对象数组进行从小到大排序,要求数组元素类型必须实现Comparable接口
  7. 5)可以给任意对象数组进行从小到大排序,只要你指定定制比较器对象,不要求数组元素实现Comparable接口
  8. 6)可以将任意对象数组的元素拼接为一个字符串返回
  9. ```java
  10. public class MyArrays {
  11. //可以在任意类型的对象数组中,查找某个元素的下标,按照顺序查找,如果有重复的,就返回第一个找到的,如果没有返回-1
  12. public static <T> int find(T[] arr, T value) {
  13. for (int i = 0; i < arr.length; i++) {
  14. if(arr[i].equals(value)) {//使用==比较太严格,使用equals方法,因为任意对象都有equals方法
  15. return i;
  16. }
  17. }
  18. return -1;
  19. }
  20. //可以在任意类型的对象数组中,查找最大值,要求元素必须实现Comparable接口
  21. public static <T extends Comparable<? super T>> T max(T[] arr) {
  22. T max = arr[0];
  23. for (int i = 0; i < arr.length; i++) {
  24. if(max.compareTo(arr[i])<0) {//if(max < arr[i]) {
  25. max = arr[i];
  26. }
  27. }
  28. return max;
  29. }
  30. //可以在任意类型的对象数组中,查找最大值,按照指定定制比较器来比较元素大小
  31. public static <T> T max(T[] arr, Comparator<? super T> c) {
  32. T max = arr[0];
  33. for (int i = 0; i < arr.length; i++) {
  34. if(c.compare(max, arr[i])<0) {//if(max < arr[i]) {
  35. max = arr[i];
  36. }
  37. }
  38. return max;
  39. }
  40. //可以给任意对象数组进行从小到大排序,要求数组元素类型必须实现Comparable接口
  41. public static <T extends Comparable<? super T>> void sort(T[] arr) {
  42. for (int i = 0; i < arr.length-1; i++) {
  43. int minIndex = i;
  44. for (int j = i+1; j < arr.length; j++) {
  45. if(arr[minIndex].compareTo(arr[j])>0) {
  46. minIndex = j;
  47. }
  48. }
  49. if(minIndex!=i) {
  50. T temp = arr[minIndex];
  51. arr[minIndex] = arr[i];
  52. arr[i] = temp;
  53. }
  54. }
  55. }
  56. //可以给任意对象数组进行从小到大排序,只要你指定定制比较器对象,不要求数组元素实现Comparable接口
  57. public static <T> void sort(T[] arr, Comparator<? super T> c) {
  58. for (int i = 0; i < arr.length-1; i++) {
  59. int minIndex = i;
  60. for (int j = i+1; j < arr.length; j++) {
  61. if(c.compare(arr[minIndex],arr[j])>0) {
  62. minIndex = j;
  63. }
  64. }
  65. if(minIndex!=i) {
  66. T temp = arr[minIndex];
  67. arr[minIndex] = arr[i];
  68. arr[i] = temp;
  69. }
  70. }
  71. }
  72. //可以将任意对象数组的元素拼接为一个字符串返回
  73. public static <T> String toString(T[] arr) {
  74. String str = "[";
  75. for (int i = 0; i < arr.length; i++) {
  76. if(i==0) {
  77. str += arr[i];
  78. }else {
  79. str += "," + arr[i];
  80. }
  81. }
  82. str += "]";
  83. return str;
  84. }
  85. }