泛型类传参小助手-泛型通配符 | 带你学《Java面向对象编程》之六十五

简介: 泛型类在有效回避转型风险的同时,也带来了传参问题,所以Java提供了类型通配符来解决这个问题。

上一篇:泛型带你规避风险 | 带你学《Java面向对象编程》之六十四
【本节目标】
通过阅读本节内容,你将发现传递泛型类参数时会出现类型指定上的问题,并了解到能够用泛型通配符来解决这个问题,学会灵活使用泛型通配符限制传参时对象的值。

泛型通配符

虽然泛型帮助开发者解决了一系列的强制转换所带来的的安全隐患,但是从另外一个角度来讲,泛型也带来了一些新的问题:引用传递处理。
范例:观察问题的产生

class Message <T> {    
    private T content ;
    public void setContent(T content) {
        this.content = content ;
    }   
    public T getContent() {
        return this.content ; 
    }
}
public class JavaDemo {
    public static void main(String args[]) {
        Message<String> msg = new Message<String>() ;
        msg.setContent(“www.mldn.cn”) ;   
        fun(msg) ;    //引用传递
    }  
    public static void fun(Message<String> temp) {
         System.out.println(temp.getContent()) ;
    }    //执行结果:www.mldn.cn
}

但是这个时候问题也就出现了,而问题的关键在于fun()方法上,如果真的去使用泛型,不可能只是一种类型,也就是说fun()方法应该可以接收任意种泛型类型的Message对象。但是这个时候它只能够接收“Message”类型,这种情况下有人提出,不设置泛型。
范例:不设置泛型

public class JavaDemo {
    public static void main(String args[]) {
        Message<Integer> msgA = new Message<Integer>() ;
        Message<String> msgB = new Message<String>() ;
        msgA.setContent(110) ;   
        fun(msgA) ;    //引用传递
        msgB.setContent(“www.mldn.cn”) ;   
        fun(msgB) ;   
    }  
    public static void fun(Message temp) {
        System.out.println(temp.getContent()) ;
    }   
}

执行结果:
110
www.mldn.cn

public static void fun(Message temp) {
    temp.setContent(1.1) ;
    System.out.println(temp.getContent()) ;
}    //执行结果:1.1   1.1

这个时候发现如果不设置泛型,那么在方法之中就有可能对数据进行修改,所以此时我们需要找一种方案,可以接收所有的泛型类型,并且不能够修改里面的数据(允许获取),那么就需要通过通配符“?”来解决。

范例:使用通配符

class Message <T> {    
    private T content ;
    public void setContent(T content) {
        this.content = content ;
    }   
    public T getContent() {
        return this.content ; 
    }
}
public class JavaDemo {
    public static void main(String args[]) {
        Message<Integer> msgA = new Message<Integer>() ;
        Message<String> msgB = new Message<String>() ;
        msgA.setContent(110) ;   
        fun(msgA) ;    //引用传递
        msgB.setContent(“www.mldn.cn”) ;   
        fun(msgB) ;   
    }  
    public static void fun(Message<?> temp) {
        System.out.println(temp.getContent()) ;
    }    
}

执行结果:
110
www.mldn.cn

此时在fun()方法里面由于采用了Message结合通配符的处理所以可以接收所有的类型,并且不允许修改,只允许获取数据。
在“?”这个通配符的基础之上实际上还提供有两类小的通配符:

  • ?extends 类:设置泛型的上限:

    • 例如:定义“?extends Number”:表示该泛型类型只允许设置Number或Number的子类;
  • ?super 类:设置泛型的下限:

    • 例如:定义“?super String”:只能够使用String或其父类;

范例:观察泛型的上限配置

class Message <T extends Number> {    
    private T content ;
    public void setContent(T content) {
        this.content = content ;
    }   
    public T getContent() {
        return this.content ; 
    }
}
public class JavaDemo {
    public static void main(String args[]) {
        Message<Integer> msgA = new Message<Integer>() ;
        Message<String> msgB = new Message<String>() ;
        msgA.setContent(110) ;   
        fun(msgA) ;    //引用传递
        msgB.setContent(“www.mldn.cn”) ;   
        fun(msgB) ;   
    }  
    public static void fun(Message<? extends Number> temp) {
        System.out.println(temp.getContent()) ;
    }    
}

image.png
图一 执行结果图

public class JavaDemo {
    public static void main(String args[]) {
        Message<Integer> msgA = new Message<Integer>() ;
        msgA.setContent(110) ;   
        fun(msgA) ;    //引用传递
    }       //执行结果:110
}

范例:设置泛型下限

class Message <T> {    
    private T content ;
    public void setContent(T content) {
        this.content = content ;
    }   
    public T getContent() {
        return this.content ; 
    }
}
public class JavaDemo {
    public static void main(String args[]) {
        Message<String> msgA = new Message<String>() ;
        msgA.setContent(“www.mldn.cn”) ;   
        fun(msgA) ;    //引用传递
    }  
    public static void fun(Message<? super String> temp) {
        System.out.println(temp.getContent()) ;
    }    //执行结果:www.mldn.cn
}

对于通配符而言,是一个重要的概念,并且要求一定要理解此概念的定义,在日后的学习Java系统类库的时候一定会见到大量的通配符使用,所以要求必须掌握。

想学习更多的Java的课程吗?从小白到大神,从入门到精通,更多精彩不容错过!免费为您提供更多的学习资源。
本内容视频来源于阿里云大学

下一篇:泛型,不只是简单类 | 带你学《Java面向对象编程》之六十六
更多Java面向对象编程文章查看此处

相关文章
|
16天前
|
安全 Java 数据建模
Java记录类:简化数据载体的新选择
Java记录类:简化数据载体的新选择
167 101
|
16天前
|
安全 Java 开发者
Java记录类:简化数据载体的新方式
Java记录类:简化数据载体的新方式
193 100
|
2月前
|
安全 IDE Java
Java记录类型(Record):简化数据载体类
Java记录类型(Record):简化数据载体类
300 120
|
9天前
|
安全 Java
Java之泛型使用教程
Java之泛型使用教程
111 10
|
2月前
|
缓存 安全 Java
Java反射机制:动态操作类与对象
Java反射机制是运行时动态操作类与对象的强大工具,支持获取类信息、动态创建实例、调用方法、访问字段等。它在框架开发、依赖注入、动态代理等方面有广泛应用,但也存在性能开销和安全风险。本文详解反射核心API、实战案例及性能优化策略,助你掌握Java动态编程精髓。
|
2月前
|
存储 安全 Java
Java集合框架(一):List接口及其实现类剖析
本文深入解析Java中List集合的实现原理,涵盖ArrayList的动态数组机制、LinkedList的链表结构、Vector与Stack的线程安全性及其不推荐使用的原因,对比了不同实现的性能与适用场景,帮助开发者根据实际需求选择合适的List实现。
|
3月前
|
Java API
深入解析Java API中Object类的功能
了解和合理运用 Object类的这些方法,对于编写可靠和高效的Java应用程序至关重要。它们构成了Java对象行为的基础,影响着对象的创建、识别、表达和并发控制。
66 0
|
3月前
|
安全 Java
JAVA:Collections类的shuffle()方法
`Collections.shuffle()` 是 Java 中用于随机打乱列表顺序的工具方法,适用于洗牌、抽奖等场景。该方法直接修改原列表,支持自定义随机数生成器以实现可重现的打乱顺序。使用时需注意其原地修改特性及非线程安全性。
136 0
|
3月前
|
存储 Java 编译器
深入理解Java虚拟机--类文件结构
本内容介绍了Java虚拟机与Class文件的关系及其内部结构。Class文件是一种与语言无关的二进制格式,包含JVM指令集、符号表等信息。无论使用何种语言,只要能生成符合规范的Class文件,即可在JVM上运行。文章详细解析了Class文件的组成,包括魔数、版本号、常量池、访问标志、类索引、字段表、方法表和属性表等,并说明其在Java编译与运行过程中的作用。