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

  类型参数的范围

  在泛型中,如果不对类型参数加以限制,它就可以接受任意的数据类型,只要它是被定义过的。但是,很多时候我们只需要一部分数据类型就够了,用户传递其他数据类型可能会引起错误。例如,编写一个泛型函数用于返回不同类型数组(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泛型通配符和类型参数的范围的更多相关文章

  1. Java知多少(42)泛型通配符和类型参数的范围

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

  2. Java学习之——泛型

    1.概要 generics enable types (classes and interfaces) to be parameters when defining classes, interfac ...

  3. Java学习之==>泛型

    一.什么是泛型 泛型,即“参数化类型”,在不创建新的类型的情况下,通过泛型指定的不同类型来控制形参具体限制的类型.也就是说在泛型使用过程中,操作的数据类型被指定为一个参数,这种参数类型可以用在类.接口 ...

  4. 【Java】java 中的泛型通配符——从“偷偷地”地改变集合元素说起

    一直没注意这方面的内容,想来这也算是基础了,就写了这个笔记. 首先java的通配符共有三种----先别紧张,现在只是粗略的过一下,看不看其实无所谓 类型 介绍 <?> 无限定通配符,等价于 ...

  5. Java学习之泛型和异常

    泛型 1,设计原则或目的:只要代码在编译的时候没有错误,就不会抛异常.  2,泛型通配符  :类型通配符一般是使用 ? 代替具体的类型实参.注意了,此处是类型实参,而不是类型形参!相当于(父类作用)L ...

  6. JAVA基础补漏--泛型通配符

    泛型通配符只能用于方法的参数 不能用对象定义 public class Test { public static void main(String[] args) { ArrayList<Str ...

  7. [Java学习] Java虚拟机(JVM)以及跨平台原理

    相信大家已经了解到Java具有跨平台的特性,可以“一次编译,到处运行”,在Windows下编写的程序,无需任何修改就可以在Linux下运行,这是C和C++很难做到的. 那么,跨平台是怎样实现的呢?这就 ...

  8. Java学习笔记--泛型

    一个泛型类就是具有一个或者多个类型变量的类. 我们可以只关注泛型,而不会为数据存储的细节而烦恼 . java泛型(一).泛型的基本介绍和使用 http://blog.csdn.net/lonelyro ...

  9. JAVA学习之泛型

    ArrayList<E>类定义和ArrayList<Integer>类引用中涉及的术语:1.整个ArrayList<E>称为泛型类型 2.ArrayList< ...

随机推荐

  1. 代码静态检查Eclipse插件:SonarLint插件离线安装

    Eclipse Version: Oxygen.3a Release (4.7.3a)Myeclipse版本: 10.7 SonarLint 插件离线安装包:org.sonarlint.eclipse ...

  2. Filter过滤器原理和登录实现

    Filter过滤器API      Servlet过滤器API包含了3个接口,它们都在javax.servlet包中,分别是Filter接口.FilterChain接口和FilterConfig接口. ...

  3. mysql 替换函数replace()实现mysql 替换字符串

    mysql 替换字符串的实现方法:mysql中replace函数直接替换mysql数据库中某字段中的特定字符串,不再需要自己写函数去替换,用起来非常的方便,mysql 替换函数replace()Upd ...

  4. 静默文件安装安装WebLogic

    一. 本文演示静默文件方式安装     •在Windows上      –打开命令行窗口 –filename.exe -mode=silent -silent_xml=file_path     •在 ...

  5. sqlserver 判断各种不存在

    判断数据库是否存在 if exists (select * from dbo.sysobjects where name = '数据库名') --drop database [数据库名] 判断表是否存 ...

  6. LOJ10067 构造完全图

    LOJ10067 构造完全图 最小生成树 每次找到最小的边,将边两端的块合并 (我之前想的是什么鬼) #include<cstdio> #include<algorithm> ...

  7. python Django编写登录项目

    Django 目录结构: __init__.py 文件: import pymysql pymysql.install_as_MySQLdb() 注意:如果 import pymysql 的时候报红, ...

  8. 20145321《网络对抗》Exp2 后门原理与实践

    实验内容 (1)使用netcat获取主机操作Shell,cron启动 (2)使用socat获取主机操作Shell, 任务计划启动 (3)使用MSF meterpreter生成可执行文件,利用ncat或 ...

  9. 20145331魏澍琛《网络对抗》Exp6 信息搜集与漏洞扫描

    20145331魏澍琛<网络对抗>Exp6 信息搜集与漏洞扫描 问题回答 1.哪些组织负责DNS,IP的管理? DNS域名服务器:绝大多数在欧洲和北美洲,中国仅拥有镜像服务器. 全球一共有 ...

  10. Android 实践项目开发二

    在地图开发中项目中,我这周主要完成的任务是和遇到的问题是以下几个方面. 1.在本次的项目中主要是利用百度地图的.jar包实现地图的定位与搜索功能,需要在百度地图开发中心网站取得 密钥,并下载相关.ja ...