泛型类传参小助手-泛型通配符 | 带你学《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面向对象编程文章查看此处

相关文章
|
5月前
|
存储 Java 索引
用Java语言实现一个自定义的ArrayList类
自定义MyArrayList类模拟Java ArrayList核心功能,支持泛型、动态扩容(1.5倍)、增删改查及越界检查,底层用Object数组实现,适合学习动态数组原理。
214 4
|
5月前
|
IDE JavaScript Java
在Java 11中,如何处理被弃用的类或接口?
在Java 11中,如何处理被弃用的类或接口?
281 5
|
5月前
|
JSON 网络协议 安全
【Java】(10)进程与线程的关系、Tread类;讲解基本线程安全、网络编程内容;JSON序列化与反序列化
几乎所有的操作系统都支持进程的概念,进程是处于运行过程中的程序,并且具有一定的独立功能,进程是系统进行资源分配和调度的一个独立单位一般而言,进程包含如下三个特征。独立性动态性并发性。
269 1
|
5月前
|
Java Go 开发工具
【Java】(8)正则表达式的使用与常用类分享
正则表达式定义了字符串的模式。正则表达式并不仅限于某一种语言,但是在每种语言中有细微的差别。
379 1
|
5月前
|
存储 Java 程序员
【Java】(6)全方面带你了解Java里的日期与时间内容,介绍 Calendar、GregorianCalendar、Date类
java.util 包提供了 Date 类来封装当前的日期和时间。Date 类提供两个构造函数来实例化 Date 对象。第一个构造函数使用当前日期和时间来初始化对象。Date( )第二个构造函数接收一个参数,该参数是从1970年1月1日起的毫秒数。
260 1
|
5月前
|
JSON 网络协议 安全
【Java基础】(1)进程与线程的关系、Tread类;讲解基本线程安全、网络编程内容;JSON序列化与反序列化
几乎所有的操作系统都支持进程的概念,进程是处于运行过程中的程序,并且具有一定的独立功能,进程是系统进行资源分配和调度的一个独立单位一般而言,进程包含如下三个特征。独立性动态性并发性。
287 1
|
5月前
|
编解码 Java 开发者
Java String类的关键方法总结
以上总结了Java `String` 类最常见和重要功能性方法。每种操作都对应着日常编程任务,并且理解每种操作如何影响及处理 `Strings` 对于任何使用 Java 的开发者来说都至关重要。
364 5
|
6月前
|
安全 Java
Java之泛型使用教程
Java之泛型使用教程
418 10
|
6月前
|
安全 Java 数据建模
Java记录类:简化数据载体的新选择
Java记录类:简化数据载体的新选择
334 101
|
6月前
|
安全 Java 开发者
Java记录类:简化数据载体的新方式
Java记录类:简化数据载体的新方式
332 100