Java泛型中通配符的使用

2024-08-28 16:08
文章标签 java 使用 泛型 通配符

本文主要是介绍Java泛型中通配符的使用,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

学习目标

掌握通配符“?” 的使用
掌握受限泛型的设置
掌握泛型与子类继承的限制

匹配任意类型的通配符

在开发中对象的引用传递是最常见的,但是如果在泛型类的操作中,在进行传递的时候泛型类型必须匹配才可以传递。否则是无法传递的。
class Info<T>{private T var ;		// 定义泛型变量public void setVar(T var){this.var = var ;}public T getVar(){return this.var ;}public String toString(){	// 直接打印return this.var.toString() ;}
};
public class GenericsDemo12{public static void main(String args[]){Info<String> i = new Info<String>() ;		// 使用String为泛型类型i.setVar("lx") ;							// 设置内容fun(i) ;}public static void fun(Info<Object> temp){		// 接收Object泛型类型的Info对象System.out.println("内容:" + temp) ;}
};

编译时就出现错误:

 泛型对象进行引用传递的时候,类型必须一致。如果现在非要传递,则可以将fun方法中的泛型取消掉。如下所示:
class Info<T>{private T var ;		// 定义泛型变量public void setVar(T var){this.var = var ;}public T getVar(){return this.var ;}public String toString(){	// 直接打印return this.var.toString() ;}
};
public class GenericsDemo13{public static void main(String args[]){Info<String> i = new Info<String>() ;		// 使用String为泛型类型i.setVar("WWWW") ;							// 设置内容fun(i) ;}public static void fun(Info temp){		// 接收Object泛型类型的Info对象System.out.println("内容:" + temp) ;}
};
发现有警告提示:

但不会影响程序的运行:

以上确实完成了改进的功能,但是,代码似乎有些不是很妥当,毕竟之前已经指定过泛型了。
如果使用?呢?如下所示:
class Info<T>{private T var ;		// 定义泛型变量public void setVar(T var){this.var = var ;}public T getVar(){return this.var ;}public String toString(){	// 直接打印return this.var.toString() ;}
};
public class GenericsDemo14{public static void main(String args[]){Info<String> i = new Info<String>() ;		// 使用String为泛型类型i.setVar("MLDN") ;							// 设置内容fun(i) ;}public static void fun(Info<?> temp){		// 可以接收任意的泛型对象System.out.println("内容:" + temp) ;}
};
能照常运行,而且不会有警告。
注意:如果使用?意味着可以接收任意的内容,但是此内容却无法直接使用<?> 修饰的泛型对象进行修改。如下所示:
class Info<T>{private T var ;		// 定义泛型变量public void setVar(T var){this.var = var ;}public T getVar(){return this.var ;}public String toString(){	// 直接打印return this.var.toString() ;}
};
public class GenericsDemo15{public static void main(String args[]){Info<?> i = new Info<String>() ;		// 使用String为泛型类型i.setVar("MLDN") ;							// 设置内容}
};

发现有错误,警告和运行结果分别如下所示:



也就是说,在使用户<?> 只能接收。但是不能修改。

受限泛型

之前设置泛型类型的时候,实际上都是可以任意设置的,只要是类就可以设置。但是在Java的泛型中可以指定一个泛型的上限和下限。范围的上限使用extends关键字声明,表示参数化的类型可能是所指定的类型,或者是此类型的子类。而范围下限使用super进行声明,表示参数化的类型可能是所指定的类型,或者是此类型的父类型,直至Object类。
设置上限:
  声明对象:类名称<? extends 类>  对象名称
  定义类: [访问权限] 类名称<泛型标示 extends 类>{}
设置下限:
  声明对象:类名称<? super 类> 对象名称
 定义类:[访问权限] 类名称<泛型标示 super 类>{}

设置上限

class Info<T>{private T var ;		// 定义泛型变量public void setVar(T var){this.var = var ;}public T getVar(){return this.var ;}public String toString(){	// 直接打印return this.var.toString() ;}
};
public class GenericsDemo17{public static void main(String args[]){Info<Integer> i1 = new Info<Integer>() ;		// 声明Integer的泛型对象Info<Float> i2 = new Info<Float>() ;			// 声明Float的泛型对象i1.setVar(30) ;									// 设置整数,自动装箱i2.setVar(30.1f) ;								// 设置小数,自动装箱fun(i1) ;fun(i2) ;}public static void fun(Info<? extends Number> temp){	// 只能接收Number及其Number的子类System.out.print(temp + "、") ;}
};


如果在方法中传递的不是泛型标示不是Number及其子类,则会报错,如下所示:
class Info<T>{private T var ;		// 定义泛型变量public void setVar(T var){this.var = var ;}public T getVar(){return this.var ;}public String toString(){	// 直接打印return this.var.toString() ;}
};
public class GenericsDemo18{public static void main(String args[]){Info<String> i1 = new Info<String>() ;		// 声明String的泛型对象i1.setVar("hello") ;fun(i1) ;}public static void fun(Info<? extends Number> temp){	// 只能接收Number及其Number的子类System.out.print(temp + "、") ;}
};

在类中也可以使用泛型,如下所示:
class Info<T extends Number>{	// 此处泛型只能是数字类型private T var ;		// 定义泛型变量public void setVar(T var){this.var = var ;}public T getVar(){return this.var ;}public String toString(){	// 直接打印return this.var.toString() ;}
};
public class GenericsDemo19{public static void main(String args[]){Info<Integer> i1 = new Info<Integer>() ;		// 声明Integer的泛型对象}
};
 如果现在在使用Info的时候设置成了String类型,则在编译的时候就会出现错误。
class Info<T extends Number>{	// 此处泛型只能是数字类型private T var ;		// 定义泛型变量public void setVar(T var){this.var = var ;}public T getVar(){return this.var ;}public String toString(){	// 直接打印return this.var.toString() ;}
};
public class GenericsDemo20{public static void main(String args[]){Info<String> i1 = new Info<String>() ;		// 声明Integer的泛型对象}
};

设置下限

当使用的泛型只能在本类以及父类类型应用的时候,就必须使用泛型的范围下限配置
class Info<T>{private T var ;		// 定义泛型变量public void setVar(T var){this.var = var ;}public T getVar(){return this.var ;}public String toString(){	// 直接打印return this.var.toString() ;}
};
public class GenericsDemo21{public static void main(String args[]){Info<String> i1 = new Info<String>() ;		// 声明String的泛型对象Info<Object> i2 = new Info<Object>() ;		// 声明Object的泛型对象i1.setVar("hello") ;i2.setVar(new Object()) ;fun(i1) ;fun(i2) ;}public static void fun(Info<? super String> temp){	// 只能接收String或Object类型的泛型System.out.print(temp + "、") ;}
};



如果现在使用了Integer作为泛型的类型,则不满足泛型的下限。如下所示:

class Info<T>{private T var ;		// 定义泛型变量public void setVar(T var){this.var = var ;}public T getVar(){return this.var ;}public String toString(){	// 直接打印return this.var.toString() ;}
};
public class GenericsDemo22{public static void main(String args[]){Info<Integer> i1 = new Info<Integer>() ;		// 声明Integer的泛型对象i1.setVar(30) ;fun(i1) ;}public static void fun(Info<? super String> temp){	// 只能接收String或Object类型的泛型System.out.print(temp + "、") ;}
};

解释:泛型与子类继承的限制
一个类的子类可以通过对象多态性,为其父类实例化,但是在泛型操作中,子类的泛型类型是无法使用父类的泛型类型接收的,例如:Info<String> 不能使用Info<Object>接收。


实例如下:
class Info<T>{private T var ;		// 定义泛型变量public void setVar(T var){this.var = var ;}public T getVar(){return this.var ;}public String toString(){	// 直接打印return this.var.toString() ;}
};
public class GenericsDemo23{public static void main(String args[]){Info<String> i1 = new Info<String>() ;		// 泛型类型为StringInfo<Object> i2 = null ;i2 = i1 ;}
};

发现有错误提示。

总结:
1、使用?可以接收任意的泛型类型。
2、泛型的上限:? extends 类型。
3、泛型的下限用的不是太多。
4、了解为什么泛型子类之间的继承无法直接转换的原因。



这篇关于Java泛型中通配符的使用的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!


原文地址:
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.chinasem.cn/article/1115245

相关文章

java实现延迟/超时/定时问题

《java实现延迟/超时/定时问题》:本文主要介绍java实现延迟/超时/定时问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Java实现延迟/超时/定时java 每间隔5秒执行一次,一共执行5次然后结束scheduleAtFixedRate 和 schedu

Java Optional避免空指针异常的实现

《JavaOptional避免空指针异常的实现》空指针异常一直是困扰开发者的常见问题之一,本文主要介绍了JavaOptional避免空指针异常的实现,帮助开发者编写更健壮、可读性更高的代码,减少因... 目录一、Optional 概述二、Optional 的创建三、Optional 的常用方法四、Optio

C++ Sort函数使用场景分析

《C++Sort函数使用场景分析》sort函数是algorithm库下的一个函数,sort函数是不稳定的,即大小相同的元素在排序后相对顺序可能发生改变,如果某些场景需要保持相同元素间的相对顺序,可使... 目录C++ Sort函数详解一、sort函数调用的两种方式二、sort函数使用场景三、sort函数排序

Spring Boot项目中结合MyBatis实现MySQL的自动主从切换功能

《SpringBoot项目中结合MyBatis实现MySQL的自动主从切换功能》:本文主要介绍SpringBoot项目中结合MyBatis实现MySQL的自动主从切换功能,本文分步骤给大家介绍的... 目录原理解析1. mysql主从复制(Master-Slave Replication)2. 读写分离3.

idea maven编译报错Java heap space的解决方法

《ideamaven编译报错Javaheapspace的解决方法》这篇文章主要为大家详细介绍了ideamaven编译报错Javaheapspace的相关解决方法,文中的示例代码讲解详细,感兴趣的... 目录1.增加 Maven 编译的堆内存2. 增加 IntelliJ IDEA 的堆内存3. 优化 Mave

Java String字符串的常用使用方法

《JavaString字符串的常用使用方法》String是JDK提供的一个类,是引用类型,并不是基本的数据类型,String用于字符串操作,在之前学习c语言的时候,对于一些字符串,会初始化字符数组表... 目录一、什么是String二、如何定义一个String1. 用双引号定义2. 通过构造函数定义三、St

springboot filter实现请求响应全链路拦截

《springbootfilter实现请求响应全链路拦截》这篇文章主要为大家详细介绍了SpringBoot如何结合Filter同时拦截请求和响应,从而实现​​日志采集自动化,感兴趣的小伙伴可以跟随小... 目录一、为什么你需要这个过滤器?​​​二、核心实现:一个Filter搞定双向数据流​​​​三、完整代码

SpringBoot利用@Validated注解优雅实现参数校验

《SpringBoot利用@Validated注解优雅实现参数校验》在开发Web应用时,用户输入的合法性校验是保障系统稳定性的基础,​SpringBoot的@Validated注解提供了一种更优雅的解... 目录​一、为什么需要参数校验二、Validated 的核心用法​1. 基础校验2. php分组校验3

Pydantic中Optional 和Union类型的使用

《Pydantic中Optional和Union类型的使用》本文主要介绍了Pydantic中Optional和Union类型的使用,这两者在处理可选字段和多类型字段时尤为重要,文中通过示例代码介绍的... 目录简介Optional 类型Union 类型Optional 和 Union 的组合总结简介Pyd

Java Predicate接口定义详解

《JavaPredicate接口定义详解》Predicate是Java中的一个函数式接口,它代表一个判断逻辑,接收一个输入参数,返回一个布尔值,:本文主要介绍JavaPredicate接口的定义... 目录Java Predicate接口Java lamda表达式 Predicate<T>、BiFuncti