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

相关文章
|
1月前
|
Java 开发者
在 Java 中,一个类可以实现多个接口吗?
这是 Java 面向对象编程的一个重要特性,它提供了极大的灵活性和扩展性。
153 57
|
4天前
|
JSON 前端开发 Java
【Bug合集】——Java大小写引起传参失败,获取值为null的解决方案
类中成员变量命名问题引起传送json字符串,但是变量为null的情况做出解释,@Data注解(Spring自动生成的get和set方法)和@JsonProperty
|
6天前
|
JSON Java Apache
Java基础-常用API-Object类
继承是面向对象编程的重要特性,允许从已有类派生新类。Java采用单继承机制,默认所有类继承自Object类。Object类提供了多个常用方法,如`clone()`用于复制对象,`equals()`判断对象是否相等,`hashCode()`计算哈希码,`toString()`返回对象的字符串表示,`wait()`、`notify()`和`notifyAll()`用于线程同步,`finalize()`在对象被垃圾回收时调用。掌握这些方法有助于更好地理解和使用Java中的对象行为。
|
1月前
|
存储 缓存 安全
java 中操作字符串都有哪些类,它们之间有什么区别
Java中操作字符串的类主要有String、StringBuilder和StringBuffer。String是不可变的,每次操作都会生成新对象;StringBuilder和StringBuffer都是可变的,但StringBuilder是非线程安全的,而StringBuffer是线程安全的,因此性能略低。
56 8
|
1月前
|
存储 安全 Java
java.util的Collections类
Collections 类位于 java.util 包下,提供了许多有用的对象和方法,来简化java中集合的创建、处理和多线程管理。掌握此类将非常有助于提升开发效率和维护代码的简洁性,同时对于程序的稳定性和安全性有大有帮助。
73 17
|
1月前
|
安全 Java
Java多线程集合类
本文介绍了Java中线程安全的问题及解决方案。通过示例代码展示了使用`CopyOnWriteArrayList`、`CopyOnWriteArraySet`和`ConcurrentHashMap`来解决多线程环境下集合操作的线程安全问题。这些类通过不同的机制确保了线程安全,提高了并发性能。
|
1月前
|
存储 Java 程序员
Java基础的灵魂——Object类方法详解(社招面试不踩坑)
本文介绍了Java中`Object`类的几个重要方法,包括`toString`、`equals`、`hashCode`、`finalize`、`clone`、`getClass`、`notify`和`wait`。这些方法是面试中的常考点,掌握它们有助于理解Java对象的行为和实现多线程编程。作者通过具体示例和应用场景,详细解析了每个方法的作用和重写技巧,帮助读者更好地应对面试和技术开发。
135 4
|
1月前
|
Java 编译器 开发者
Java异常处理的最佳实践,涵盖理解异常类体系、选择合适的异常类型、提供详细异常信息、合理使用try-catch和finally语句、使用try-with-resources、记录异常信息等方面
本文探讨了Java异常处理的最佳实践,涵盖理解异常类体系、选择合适的异常类型、提供详细异常信息、合理使用try-catch和finally语句、使用try-with-resources、记录异常信息等方面,帮助开发者提高代码质量和程序的健壮性。
78 2
|
1月前
|
存储 安全 Java
如何保证 Java 类文件的安全性?
Java类文件的安全性可以通过多种方式保障,如使用数字签名验证类文件的完整性和来源,利用安全管理器和安全策略限制类文件的权限,以及通过加密技术保护类文件在传输过程中的安全。
61 4
|
1月前
|
Java Android开发
Eclipse 创建 Java 类
Eclipse 创建 Java 类
29 0