本节先讲解如何限制类型参数的范围,再讲解通配符(?)。

类型参数的范围

在泛型中,如果不对类型参数加以限制,它就可以接受任意的数据类型,只要它是被定义过的。但是,很多时候我们只需要一部分数据类型就够了,用户传递其他数据类型可能会引起错误。例如,编写一个泛型函数用于返回不同类型数组(Integer 数组、Double 数组等)中的最大值:

  1. public <T> 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 extends Number> 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 extends Number> 表示 T 只接受 Number 及其子类,传入其他类型的数据会报错。这里的限定使用关键字 extends,后面可以是类也可以是接口。如果是类,只能有一个;但是接口可以有多个,并以“&”分隔,例如 <T extends Interface1 & Interface2>。

这里的 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. class Point<T1, T2>{
  19. T1 x;
  20. T2 y;
  21. public T1 getX() {
  22. return x;
  23. }
  24. public void setX(T1 x) {
  25. this.x = x;
  26. }
  27. public T2 getY() {
  28. return y;
  29. }
  30. public void setY(T2 y) {
  31. this.y = y;
  32. }
  33. }

运行结果:

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<? extends Number, ? extends Number> p){
  16. System.out.println("x: " + p.getX() + ", y: " + p.getY());
  17. }
  18.  
  19. public static void printStrPoint(Point<? extends String, ? extends String> p){
  20. System.out.println("GPS: " + p.getX() + "," + p.getY());
  21. }
  22. }
  23. class Point<T1, T2>{
  24. T1 x;
  25. T2 y;
  26. public T1 getX() {
  27. return x;
  28. }
  29. public void setX(T1 x) {
  30. this.x = x;
  31. }
  32. public T2 getY() {
  33. return y;
  34. }
  35. public void setY(T2 y) {
  36. this.y = y;
  37. }
  38. }

运行结果:

x: 10, y: 20
GPS: 东京180度,北纬210度

? extends Number 表示泛型的类型参数只能是 Number 及其子类,? extends String 也一样,这与定义泛型类或泛型方法时限制类型参数的范围类似。

不过,使用通配符(?)不但可以限制类型的上限,还可以限制下限。限制下限使用 super 关键字,例如 <? super Number> 表示只能接受 Number 及其父类。

注意:一般的项目中很少会去设计泛型,这里主要是让读者学会如何使用,为后面的教程做铺垫。

Java知多少(42)泛型通配符和类型参数的范围的更多相关文章

  1. [Java学习] java泛型通配符和类型参数的范围

    本节先讲解如何限制类型参数的范围,再讲解通配符(?). 类型参数的范围 在泛型中,如果不对类型参数加以限制,它就可以接受任意的数据类型,只要它是被定义过的.但是,很多时候我们只需要一部分数据类型就够了 ...

  2. Java那些事-泛型通配符

    Java的类型通配符,可以出现在类.方法上面.最常用的方式就是集合类,例如List,Set等类上面. 通配符类型 有泛型参数 List 有无类型标识 List< ? > 有通用的标识 Li ...

  3. Java知多少(完结篇)

    Java知多少(1)语言概述 Java知多少(2)虚拟机(JVM)以及跨平台原理 Java知多少(3) 就业方向 Java知多少(4)J2SE.J2EE.J2ME的区别 Java知多少(5) Java ...

  4. Java知多少(中)

    Java知多少(上) )interface接口 )接口和抽象类的区别 )泛型详解 )泛型通配符和类型参数的范围 )异常处理基础 )异常类型 )未被捕获的异常 )try和catch的使用 )多重catc ...

  5. [转]JAVA泛型通配符T,E,K,V区别,T以及Class<T>,Class<?>的区别

    原文地址:https://www.jianshu.com/p/95f349258afb 1. 先解释下泛型概念 泛型是Java SE 1.5的新特性,泛型的本质是参数化类型,也就是说所操作的数据类型被 ...

  6. JAVA泛型通配符T,E,K,V区别,T以及Class<T>,Class<?>的区别

    1. 先解释下泛型概念 泛型是Java SE 1.5的新特性,泛型的本质是参数化类型,也就是说所操作的数据类型被指定为一个参数.这种参数类型可以用在类.接口和方法的创建中,分别称为泛型类.泛型接口.泛 ...

  7. -1-3 java集合框架基础 java集合体系结构 Collection 常用java集合框架 如何选择集合 迭代器 泛型 通配符概念 Properties 集合 迭代器

    集合又称之为容器存储对象的一种方式 •数组虽然也可以存储对象,但长度是固定的:显然需要可变长度的容器 集合和数组的区别?                 A:长度区别                  ...

  8. Java知多少(41)泛型详解

    我们知道,使用变量之前要定义,定义一个变量时必须要指明它的数据类型,什么样的数据类型赋给什么样的值. 假如我们现在要定义一个类来表示坐标,要求坐标的数据类型可以是整数.小数和字符串,例如: x = 1 ...

  9. Java 之泛型通配符 ? extends T 与 ? super T 解惑

    简述 大家在平时的工作学习中, 肯定会见过不少如下的语句: List<? super T> List<? extends T> 我们都知道, 上面的代码时关于 Java 泛型的 ...

随机推荐

  1. 潭州课堂25班:Ph201805201 WEB 之 HTML 第一课 (课堂笔记)

    什么是HTML 超文本标记语言(HyperText Markup Language,简称 HTML) HTML 是一门标记语言,标记语言由一套标记标签组成, 学习 HTML,其实就是学习标签 HTML ...

  2. js正则表达式中的

    $(function(){ //match方法---返回的是数组 var str = "1215rere565656"; alert(str.match(/\d+/g));cons ...

  3. linux命令(及解压tar.gz文件)

    https://wenku.baidu.com/view/f5805017866fb84ae45c8df3.html 1.压缩命令: 命令格式:tar  -zcvf   压缩文件名.tar.gz   ...

  4. [ Visual Studio ] MSDN

    在 Visual Studio 中创建自定义项目和项模板 编写和重构代码 (C++) C# 指南 C#最新版本 使用 MSBuild 如何:管理编辑器模式,进入全屏模式编写代码 自定义代码折叠

  5. 前端AngularJS后端ASP.NET Web API上传文件

    本篇体验使用AngularJS向后端ASP.NET API控制器上传文件.    首先服务端: public class FilesController : ApiController { //usi ...

  6. Java并发(理论知识)—— 线程安全性

    1.什么是线程安全性                                                                                      当多个线 ...

  7. Image Processing in Python with Pillow

    Introduction A lot of applications use digital images, and with this there is usually a need to proc ...

  8. 【POJ 3694】 Network(割边&lt;桥&gt;+LCA)

    [POJ 3694] Network(割边+LCA) Network Time Limit: 5000MS   Memory Limit: 65536K Total Submissions: 7971 ...

  9. 内联汇编中的asm和__asm__

    基本的内联汇编代码: asm格式: asm("assembly code"):   使用替换的关键字: 如果必须的话,可以改变用于标识内联汇编代码段的关键字asm.ANSI C规范 ...

  10. 2D游戏新手引导点光源和类迷雾实现

    一.新手引导须要的遮罩效果 一般做新手引导的时候,会把游戏画面变的半黑,然后须要玩家点击的地方就亮起来.经常使用的做法是採用遮罩来实现,可是仅仅能实现方形的,不能不规则图形.以及是全然挖空.做不到渐变 ...