[Java学习] java泛型通配符和类型参数的范围
本节先讲解如何限制类型参数的范围,再讲解通配符(?)。
类型参数的范围
在泛型中,如果不对类型参数加以限制,它就可以接受任意的数据类型,只要它是被定义过的。但是,很多时候我们只需要一部分数据类型就够了,用户传递其他数据类型可能会引起错误。例如,编写一个泛型函数用于返回不同类型数组(Integer 数组、Double 数组等)中的最大值:
1. public T getMax(T array[]){
2. T max = null;
3. for(T element : array){
4. max = element.doubleValue() > max.doubleValue() ? element : max;
5. }
6. return max;
7. }
上面的代码会报错,doubleValue() 是 Number 类及其子类的方法,不是所有的类都有该方法,所以我们要限制类型参数 T,让它只能接受 Number 及其子类(Integer、Double、Character 等)。
通过 extends 关键字可以限制泛型的类型的上限,改进上面的代码:
1. public T getMax(T array[]){
2. T max = null;
3. for(T element : array){
4. max = element.doubleValue() > max.doubleValue() ? element : max;
5. }
6. return max;
7. }
表示 T 只接受 Number 及其子类,传入其他类型的数据会报错。这里的限定使用关键字 extends,后面可以是类也可以是接口。如果是类,只能有一个;但是接口可以有多个,并以“&”分隔,例如 。
这里的 extends 关键字已不再是继承的含义了,应该理解为 T 是继承自 Number 类的类型,或者 T 是实现了 XX 接口的类型。
通配符(?)
上一节的例子中提到要定义一个泛型类来表示坐标,坐标可以是整数、小数或字符串,请看下面的代码:
1. class Point<t1, t2="">{
2. T1 x;
3. T2 y;
4. public T1 getX() {
5. return x;
6. }
7. public void setX(T1 x) {
8. this.x = x;
9. }
10. public T2 getY() {
11. return y;
12. }
13. public void setY(T2 y) {
14. this.y = y;
15. }
16. }
现在要求在类的外部定义一个 printPoint() 方法用于输出坐标,怎么办呢?
可以这样来定义方法:
1. public void printPoint(Point p){
2. System.out.println("This point is: " + p.getX() + ", " + p.getY());
3. }
我们知道,如果在使用泛型时没有指名具体的数据类型,就会擦除泛型类型,并向上转型为 Object,这与不使用泛型没什么两样。上面的代码没有指明数据类型,相当于:
1. public void printPoint(Point<object, object=""> p){
2. System.out.println("This point is: " + p.getX() + ", " + p.getY());
3. }
为了避免类型擦除,可以使用通配符(?):
1. public void printPoint(Point p){
2. System.out.println("This point is: " + p.getX() + ", " + p.getY());
3. }
通配符(?)可以表示任意的数据类型。将代码补充完整:
1. public class Demo {
2. public static void main(String[] args){
3. Point<integer, integer=""> p1 = new Point<integer, integer="">();
4. p1.setX(10);
5. p1.setY(20);
6. printPoint(p1);
7.
8. Point<string, string=""> p2 = new Point<string, string="">();
9. p2.setX("东京180度");
10. p2.setY("北纬210度");
11. printPoint(p2);
12. }
13.
14. public static void printPoint(Point p){ // 使用通配符
15. System.out.println("This point is: " + p.getX() + ", " + p.getY());
16. }
17. }
18.
19. class Point<t1, t2="">{
20. T1 x;
21. T2 y;
22. public T1 getX() {
23. return x;
24. }
25. public void setX(T1 x) {
26. this.x = x;
27. }
28. public T2 getY() {
29. return y;
30. }
31. public void setY(T2 y) {
32. this.y = y;
33. }
34. }
运行结果:
This point is: 10, 20
This point is: 东京180度, 北纬210度
但是,数字坐标与字符串坐标又有区别:数字可以表示x轴或y轴的坐标,字符串可以表示地球经纬度。现在又要求定义两个方法分别处理不同的坐标,一个方法只能接受数字类型的坐标,另一个方法只能接受字符串类型的坐标,怎么办呢?
这个问题的关键是要限制类型参数的范围,请先看下面的代码:
1. public class Demo {
2. public static void main(String[] args){
3. Point<integer, integer=""> p1 = new Point<integer, integer="">();
4. p1.setX(10);
5. p1.setY(20);
6. printNumPoint(p1);
7.
8. Point<string, string=""> p2 = new Point<string, string="">();
9. p2.setX("东京180度");
10. p2.setY("北纬210度");
11. printStrPoint(p2);
12. }
13.
14. // 借助通配符限制泛型的范围
15. public static void printNumPoint(Point p){
16. System.out.println("x: " + p.getX() + ", y: " + p.getY());
17. }
18.
19. public static void printStrPoint(Point p){
20. System.out.println("GPS: " + p.getX() + "," + p.getY());
21. }
22. }
23.
24. class Point<t1, t2="">{
25. T1 x;
26. T2 y;
27. public T1 getX() {
28. return x;
29. }
30. public void setX(T1 x) {
31. this.x = x;
32. }
33. public T2 getY() {
34. return y;
35. }
36. public void setY(T2 y) {
37. this.y = y;
38. }
39. }
运行结果:
x: 10, y: 20
GPS: 东京180度,北纬210度
? extends Number 表示泛型的类型参数只能是 Number 及其子类,? extends String 也一样,这与定义泛型类或泛型方法时限制类型参数的范围类似。
不过,使用通配符(?)不但可以限制类型的上限,还可以限制下限。限制下限使用 super 关键字,例如 表示只能接受 Number 及其父类。(编辑:雷林鹏 来源:网络)
[Java学习] java泛型通配符和类型参数的范围的更多相关文章
- Java知多少(42)泛型通配符和类型参数的范围
本节先讲解如何限制类型参数的范围,再讲解通配符(?). 类型参数的范围 在泛型中,如果不对类型参数加以限制,它就可以接受任意的数据类型,只要它是被定义过的.但是,很多时候我们只需要一部分数据类型就够了 ...
- Java学习之——泛型
1.概要 generics enable types (classes and interfaces) to be parameters when defining classes, interfac ...
- Java学习之==>泛型
一.什么是泛型 泛型,即“参数化类型”,在不创建新的类型的情况下,通过泛型指定的不同类型来控制形参具体限制的类型.也就是说在泛型使用过程中,操作的数据类型被指定为一个参数,这种参数类型可以用在类.接口 ...
- 【Java】java 中的泛型通配符——从“偷偷地”地改变集合元素说起
一直没注意这方面的内容,想来这也算是基础了,就写了这个笔记. 首先java的通配符共有三种----先别紧张,现在只是粗略的过一下,看不看其实无所谓 类型 介绍 <?> 无限定通配符,等价于 ...
- Java学习之泛型和异常
泛型 1,设计原则或目的:只要代码在编译的时候没有错误,就不会抛异常. 2,泛型通配符 :类型通配符一般是使用 ? 代替具体的类型实参.注意了,此处是类型实参,而不是类型形参!相当于(父类作用)L ...
- JAVA基础补漏--泛型通配符
泛型通配符只能用于方法的参数 不能用对象定义 public class Test { public static void main(String[] args) { ArrayList<Str ...
- [Java学习] Java虚拟机(JVM)以及跨平台原理
相信大家已经了解到Java具有跨平台的特性,可以“一次编译,到处运行”,在Windows下编写的程序,无需任何修改就可以在Linux下运行,这是C和C++很难做到的. 那么,跨平台是怎样实现的呢?这就 ...
- Java学习笔记--泛型
一个泛型类就是具有一个或者多个类型变量的类. 我们可以只关注泛型,而不会为数据存储的细节而烦恼 . java泛型(一).泛型的基本介绍和使用 http://blog.csdn.net/lonelyro ...
- JAVA学习之泛型
ArrayList<E>类定义和ArrayList<Integer>类引用中涉及的术语:1.整个ArrayList<E>称为泛型类型 2.ArrayList< ...
随机推荐
- Java互联网架构-Mysql分库分表订单生成系统实战分析
概述 分库分表的必要性 首先我们来了解一下为什么要做分库分表.在我们的业务(web应用)中,关系型数据库本身比较容易成为系统性能瓶颈,单机存储容量.连接数.处理能力等都很有限,数据库本身的“有状态性” ...
- MFC六大核心机制之四:永久保存(串行化)
永久保存(串行化)是MFC的重要内容,可以用一句简明直白的话来形容其重要性:弄懂它以后,你就越来越像个程序员了! 如果我们的程序不需要永久保存,那几乎可以肯定是一个小玩儿.那怕我们的记事本.画图等小程 ...
- 数据仓库基础(七)Informatica PowerCenter介绍
本文转载自:http://www.cnblogs.com/evencao/p/3140938.html Infromatica PowerCenter介绍: 1993年在美国加利福尼亚州成立,一年后 ...
- php json_decode() 如果想要强制生成PHP关联数组,json_decode()需要加一个参数true
php json_decode()该函数用于将json文本转换为相应的PHP数据结构.下面是一个例子:$json = '{"foo": 12345}';$obj = json_de ...
- linux服务器文件索引inodes满了
inode节点中,记录了文件的类型.大小.权限.所有者.文件连接的数目.创建时间与更新时间等重要的信息,还有一个比较重要的内容就是指向数据块的指针.一般情况不需要特殊配置,如果存放文件很多,需要配置. ...
- 【前端开发】利用Fiddler抓包工具进行本地调试
解决什么问题: 解决前端在本地联调页面 || 样式 || 脚本时经常修改服务器代码,浪费太多时间. 避免多人同时修改代码产生冲突问题.可以在本地调完代码之后,再贴到服务器上. 其实这个问题老早就开始想 ...
- SNMP学习笔记之SNMPv3的报文格式以及基于USM的认证和加密过程
下面我们就主要讲解SNMPv3的报文格式以及基于USM的认证和加密过程! 1.SNMPv3的消息格式 如下图1: 图 1 其中,整个SNMPv3消息可以使用认证机制,并对EngineID.Contex ...
- Python Web学习笔记之TCP的3次握手与4次挥手过程
前言 尽管TCP和UDP都使用相同的网络层(IP),TCP却向应用层提供与UDP完全不同的服务.TCP提供一种面向连接的.可靠的字节流服务. 面向连接意味着两个使用TCP的应用(通常是一个客户和一个服 ...
- jQuery 概述
jQuery 概述 版权声明:未经博主授权,内容严禁分享转载! 什么是 JavaScript 类库 JavaScript 类库是指已经被封装好的一系列 JavaScript 函数,能够实现一些特定的功 ...
- HTML基本格式
<html> <head> <title>放置文章标题</title> <meta http-equiv="Content-Type&q ...