泛型进阶: 泛型方法 & 通配符

偏执的太偏执、 2024-02-05 23:43 147阅读 0赞

泛型方法

定义语法

方法限定符 <类型参数列表> 返回值类型 方法名称(形参列表) {…}

示例

  1. public class Test {
  2. //静态的泛型方法 需要在static后用<>声明泛型参数
  3. public static <E> void swap(E[] array, int i, int j) {
  4. E t = array[i];
  5. array[i] = array[j];
  6. array[j] = t;
  7. }
  8. public static void main(String[] args) {
  9. //完成一个字符数组的倒转,注意这里的类型是包装类
  10. Character[] arr = {'h', 'e', 'l', 'l', 'o'};
  11. for(int i = 0; i < arr.length / 2; i++) {
  12. swap(arr, i, arr.length - 1 - i);
  13. }
  14. System.out.println(Arrays.toString(arr));
  15. }
  16. }

运行结果:

83b24628dd154952993e12916713d10c.png

通配符

? 用于在泛型的使用,即为通配符

通配符解决什么问题

  1. class Message<T> {
  2. private T message;
  3. public T getMessage() {
  4. return message;
  5. }
  6. public void setMessage(T message) {
  7. this.message = message;
  8. }
  9. }
  10. public class Test1 {
  11. public static void main(String[] args) {
  12. Message<String> message = new Message<>();
  13. message.setMessage("我不吃牛肉");
  14. fun(message);
  15. }
  16. public static void fun(Message<String> temp) {
  17. System.out.println(temp.getMessage());
  18. }
  19. }

以上程序会带来新的问题,如果现在泛型的类型设置不是String, 而是Integer.

  1. public class Test1 {
  2. public static void main(String[] args) {
  3. Message<String> message = new Message<>();
  4. message.setMessage(99);
  5. fun(message);
  6. }
  7. public static void fun(Message<String> temp) {
  8. System.out.println(temp.getMessage());
  9. }
  10. }

我们需要的解决方案: 可以接收所有的泛型类型, 但又不能让用户随意修改.这种情况就需要使用通配符”?”来处理

范例:使用通配符

  1. public class Test1 {
  2. public static void main(String[] args) {
  3. Message<String> message = new Message<>();
  4. message.setMessage(99);
  5. fun(message);
  6. }
  7. //此时使用通配符"?"描述的是它可以接收任意类型,但是由于不确定类型,所以无法修改
  8. public static void fun(Message<?> temp) {
  9. //temp.setMessage(100);无法修改!
  10. System.out.println(temp.getMessage());
  11. }
  12. }

在”?”的基础上又产生了两个子通配符:

? extends 类: 设置通配符上限

? super 类: 设置了通配符下限

通配符上界

语法:

<? extends 上界>

<? extends Number> //可以传入的实参类型是Number或者Number的子类

举个栗子:

e8d7d3ae815343c6bbc0fa63e9426032.png

  1. class Food {
  2. }
  3. class Fruit extends Food {
  4. }
  5. class Apple extends Fruit {
  6. }
  7. class Banana extends Fruit {
  8. }
  9. class Message<T> {//设置泛型
  10. private T message;
  11. public T getMessage() {
  12. return message;
  13. }
  14. public void setMessage(T message) {
  15. this.message = message;
  16. }
  17. }
  18. public class Test2 {
  19. public static void main(String[] args) {
  20. Message<Apple> message = new Message<>();
  21. message.setMessage(new Apple());
  22. fun(message);
  23. Message<Apple> message2 = new Message<>();
  24. message2.setMessage(new Banana());
  25. fun(message2);
  26. }
  27. //此时使用通配符"?"描述的是它可以接收任意类型, 但是由于不确定类型, 所以无法修改
  28. public static void fun(Message<? extends Fruit> temp) {
  29. //temp.setMessage(new Banana()); //仍然无法修改
  30. //temp.setMessage(new Apple()); //仍然无法修改
  31. System.out.println(temp.getMessage());
  32. }
  33. }

此时无法在fun函数中对temp进行添加元素,因为temp接收的是Fruit和它的子类, 此时存储的元素应该是哪个子类无法确定.所以会添加报错!但是仍然可以获取元素.

  1. //此时使用通配符"?"描述的是它可以接收任意类型, 但是由于不确定类型, 所以无法修改
  2. public static void fun(Message<? extends Fruit> temp) {
  3. //temp.setMessage(new Banana()); //仍然无法修改
  4. //temp.setMessage(new Apple()); //仍然无法修改
  5. Fruit b = temp.getMessage();
  6. System.out.println(b);
  7. }

通配符的上界, 不能写入数据, 只能进行读取数据.

通配符的下界

语法:

<? super 下界>

<? super Integer> //代表可以传入的实参的类型是Integer或者Integer的父类类型

06e72a9a49524653b768800712dfb0a7.png

  1. class Food {}
  2. class Fruit extends Food {}
  3. class Apple extends Fruit {}
  4. class Plate<T> {
  5. private T plate;
  6. public T getPlate() {
  7. return plate;
  8. }
  9. public void setPlate(T plate) {
  10. this.plate = plate;
  11. }
  12. }
  13. public class Test {
  14. public static void main(String[] args) {
  15. }
  16. public static void fun(Plate<? super Fruit> temp) {
  17. //此时可以修改!添加的是Fruit或者是Fruit的子类
  18. temp.setPlate(new Apple());//这个是Fruit的子类
  19. temp.setPlate(new Fruit());//这个是Fruit本身
  20. //Fruit fruit = temp.getPlate(); 不能直接接收,这里无法确定是哪个父类
  21. System.out.println(temp.getPlate());//只能直接输出
  22. }
  23. }

通配符的下界,不能进行读取数据,只能写入数据.

发表评论

表情:
评论列表 (有 0 条评论,147人围观)

还没有评论,来说两句吧...

相关阅读

    相关 通配符

    > 在看Java核心技术卷I中泛型通配符相关的知识时产生了很多疑问,通过了解后简单的做下笔记方便回顾。 > 本文重点记录了一下这几个问题: > > 为什么要用泛型通