Java泛型中<?>和<T>的区别浅析

目录
  • 一、定义
    • 1、T 代表一种类型
    • 2、?是通配符,泛指所有类型
  • 二、使用
    • 1、T 一般有两种用途
    • 2、<?> 的限制用途
    • 3、三种泛型限定
  • 三、总结
    • 1、从定义上看
    • 2、从用途上看
  • 补充:场景

一、定义

1、T 代表一种类型

可以加在类上,也可以加在方法上

1)T 加在类上

class SuperClass<A>{
	//todo
}

2)T 加在方法上

public <T>void fromArrayToList(T[] arr, List<T> list){
 	//todo
}

说明:

> 上面方法上的<T>代表括号里面要用到泛型参数,若类中传了泛型,此处可以不传,
> 调用类型上面的泛型参数,前提是方法中使用的泛型与类中传来的泛型一致,
  如下所示:
  /**
 * 类上的 泛型T 和 方法中的参数相同,
 * 所以方法的返回值处 可以 不加<T>
 */
class SuperClass<T>{

  public void fromArrayToList(T[] arr, List<T> list){
    //todo
  }

}

扩展:

如果有泛型方法和非泛型方法,都满足条件,会执行非泛型方法

	//非泛型方法
	public void show(String s){

      System.out.println("1");

   }

	//泛型方法
   @Override
   public void show(T a) {

      System.out.println("2");

   }

如上示例,调用show方法时,因为非泛型方法存在,会优先执行show(String s)方法

2、?是通配符,泛指所有类型

一般用于定义一个引用变量

SuperClass<?> sup = new SuperClass<String>("lisi");

sup = new SuperClass<People>(new People());

sup = new SuperClass<Animal>(new Animal());

若不用?,用固定的类型的话,则:

SuperClass<String> sup1 = new SuperClass<String>("lisi");

SuperClass<People> sup2 = new SuperClass<People>;

SuperClass<Animal> sup3 = new SuperClass<Animal>;

这就是 ? 通配符的好处:定义一个sup的引用变量,就可以指向多个对象。

二、使用

1、T 一般有两种用途

1) 定义一个通用的泛型方法

// Dao 定义1个通用的泛型方法getList
public interface Dao{
  List<T> getList(){};
}

// 使用 Dao的泛型方法getList 返回String类型
List<String> getStringList(){
  return dao.getList();//dao是一个实现类实例
}

// 使用 Dao的泛型方法getList 返回Integer类型
List<Integer> getIntList(){
  return dao.getList();
}

PS:上面接口的getList方法如果定义成List<?> ,后面就会报错。

2) 限制方法的参数之间或参数和返回结果之间的关系

// 限制参数 param1、param2、返回值都是T类型
List<T> getList<T param1,T param2>

这样可以限制返回结果的类型以及两个参数的类型一致

2、<?> 的限制用途

<?> 一般就是在泛型起一个限制作用,如下示例:

// 定义1个父类Fruit
public Class Fruit(){}

// 定义1个子类Apple,继承Fruit
public Class Apple extends Fruit(){}

// 定义1个方法test,限制参数类型只能是 Fruit 或 Fruit的子类
public void test(? extends Fruit){
	//todo
};

/** 以下是调用 */
test(new Fruit()); // 传参为 父类Fruit

test(new Apple()); // 传参为 父类Fruit的子类Apple

test(new String()); //这个会报错, 因为定义方法的传参,限制死了

3、三种泛型限定

ArrayList<T> al=new ArrayList<T>(); //指定集合元素只能是T类型
ArrayList<? extends E> al=new ArrayList<? extends E>(); //指定集合元素只能是E类型或者E的子类型
ArrayList<? super E> al=new ArrayList<? super E>(); //指定集合元素只能是E类型或者E的父类型

三、总结

1、从定义上看

T和?定义的地方有点不同,?是定义在引用变量上,T是类上或方法上

2、从用途上看

“T>“和”<?>”,首先要区分开两种不同的场景:

第一,声明一个泛型类或泛型方法;

第二,使用泛型类或泛型方法。

类型参数“T>”主要用于第一种,声明泛型类或泛型方法

无界通配符“<?>”主要用于第二种,使用泛型类或泛型方法

补充:场景

ArrayList al=new ArrayList();指定集合元素只能是T类型
ArrayList<?> al=new ArrayList<?>();集合元素可以是任意类型,这种没有意义,一般是方法中,只是为了说明用法
ArrayList<? extends E> al=new ArrayList<? extends E>();

泛型的限定:

  • ? extends E:接收E类型或者E的子类型。
  • ?super E:接收E类型或者E的父类型。

到此这篇关于Java泛型中<?>和<T>区别的文章就介绍到这了,更多相关Java泛型&lt;?&gt;和&lt;T&gt;区别内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • 浅谈三分钟学习Java泛型中T、E、K、V、?的含义

    泛型是Java中一个非常重要的内容,对于Java进阶学习是必须要掌握的知识点之所以说这个知识点重要,如果你有过阅读过一些开源框架的代码,那你一定会看到源码中有很多地方使用到了泛型. 随便举两个例子,一个List,一个Map. 看了上面的源码,简单聊一下泛型,也就是回顾一下泛型的相关知识,来源百度百科. [泛型是Java SE 1.5的新特性,泛型的本质是参数化类型,也就是说所操作的数据类型被指定为一个参数.这种参数类型可以用在类.接口和方法的创建中,分别称为泛型类.泛型接口.泛型方法.Java语

  • Java 中泛型 T 和 ? 的区别详解

    目录 泛型中 T 类型变量 和 ? 通配符 区别 Generic Types 类型变量 用法 2.声明通用的方法 – 泛型方法: 有界类型参数 Wildcards 通配符 1.上界通配符:? extend 上界类型 2.无界通配符:? 3.下界通配符:? super 子类 类型擦除 泛型中 T 类型变量 和 ? 通配符 区别 定义不同 :T 是类型变量,? 是通配符 使用范围不同: ? 通配符用作 参数类型.字段类型.局部变量类型,有时作为返回类型(但请避免这样做) T 用作 声明类的类型参数.

  • Java泛型T,E,K,V,N,?与Object区别和含义

    目录 什么是泛型 泛型方法 语法规则 泛型标记符 泛型类 类型通配符 ? ?extendsT ?superT T和? T和Object 总结 通常我们在看一些源码时,发现全是T.?,晕乎乎的:sob:.于是,把泛型掌握好十分重要! 什么是泛型 Java 泛型(generics)是 JDK 5 中引入的一个新特性, 泛型提供了编译时类型安全检测机制,该机制允许程序员在编译时检测到非法的类型. 泛型的本质是参数化类型,也就是说所操作的数据类型被指定为一个参数. 泛型有什么好处?写个例子一目了然: 我

  • java泛型中占位符T和?有什么区别

    泛型中占位符T和?有什么区别?这是一个好问题,有的人可能弄不清楚,所以我们这里简单的演示一下,相信大家一定能弄清楚的! 先上两段代码: public static <T> void show1(List<T> list){ for (Object object : list) { System.out.println(object.toString()); } } public static void show2(List<?> list) { for (Object

  • 详谈Java中的Object、T(泛型)、?区别

    因为最近重新看了泛型,又看了些反射,导致我对Object.T(以下代指泛型).?产生了疑惑. 我们先来试着理解一下Object类,学习Java的应该都知道Object是所有类的父类,注意:那么这就意味着它的范围非常广!首先记住这点,如果你的参数类型时Object,那么的参数类型将非常广! <Thinking in Java>中说很多原因促成了泛型的出现,最引人注目的一个原因就是为了创造容器类.这个要怎么来理解呢?我的理解是,可以抛开这个为了创造容器类这个,而是回到泛型的目的是限定某种类型上来.

  • Java泛型中<?>和<T>的区别浅析

    目录 一.定义 1.T 代表一种类型 2.?是通配符,泛指所有类型 二.使用 1.T 一般有两种用途 2.<?> 的限制用途 3.三种泛型限定 三.总结 1.从定义上看 2.从用途上看 补充:场景 一.定义 1.T 代表一种类型 可以加在类上,也可以加在方法上 1)T 加在类上 class SuperClass<A>{ //todo } 2)T 加在方法上 public <T>void fromArrayToList(T[] arr, List<T> lis

  • 详谈Java泛型中T和问号(通配符)的区别

    类型本来有:简单类型和复杂类型,引入泛型后把复杂类型分的更细了. 概述 泛型是Java SE 1.5的新特性,泛型的本质是参数化类型,也就是说所操作的数据类型被指定为一个参数.这种参数类型可以用在类.接口和方法的创建中,分别称为泛型类.泛型接口.泛型方法. Java语言引入泛型的好处是安全简单. 在Java SE 1.5之前,没有泛型的情况的下,通过对类型Object的引用来实现参数的"任意化","任意化"带来的缺点是要做显式的强制类型转换,而这种转换是要求开发者对

  • 最新Java 泛型中的通配符讲解

    目录 一.什么是类型擦除? 二.案例实体准备 三.常用的 ?, T, E, K, V, N的含义 四.上界通配符 < ? extends E> 五.下界通配符 < ? super E> 六.什么是PECS原则? 七.通过一个案例来理解 ?和 T 和 Object 的区别 本文内容如下: 1. 什么是类型擦除2.常用的 ?, T, E, K, V, N的含义3.上界通配符 < ?extends E>4.下界通配符 < ?super E>5.什么是PECS原则6

  • 详解Java泛型中类型擦除问题的解决方法

    以前就了解过Java泛型的实现是不完整的,最近在做一些代码重构的时候遇到一些Java泛型类型擦除的问题,简单的来说,Java泛型中所指定的类型在编译时会将其去除,因此List 和 List 在编译成字节码的时候实际上是一样的.因此java泛型只能做到编译期检查的功能,运行期间就不能保证类型安全.我最近遇到的一个问题如下: 假设有两个bean类 /** Test. */ @Data @NoArgsConstructor @AllArgsConstructor public static class

  • Java程序与C语言的区别浅析

    Java和C语言虽有相同性,但两者也有一定的不同.Java程序是面向对象的一种简单.分布式 .解释.健壮.安全.结构中立.可移植.高效能.多线程.动态的语言它是面向对象而C语言是面向过程的,这是最大的不同,对于学过C语言的我们来说,Java可以说是比较简单的编程语言. 在Java语言中,所有的变量必须要首先声明,然后才能使用. Java中可以有若干类但有且只有一个public,一个类中至少含有一个main方法,否则程序不能运行.而一个C语言由若干函数构成,其中必须且只有一个main的主函数,程序

  • Java Mybatis中的 ${ } 和 #{ }的区别使用详解

    好了,真正做开发也差不多一年了.一直都是看别人的博客,自己懒得写,而且也不会写博客,今天就开始慢慢的练习一下写博客吧.前段时间刚好在公司遇到这样的问题. 一.举例说明 select * from user where name = "dato"; select * from user where name = #{name}; select * from user where name = '${name}'; 一般情况下,我们都不会注意到这里面有什么不一样的地方.因为这些sql都可以

  • Spring中@Validated和@Valid区别浅析

    目录 基本概念 @Valid和@Validated 批注 主要区别 什么是嵌套验证? 总结 基本概念 Spring Validation 验证框架对参数的验证机制提供了@Validated(Spring's JSR-303规范,是标准JSR-303的一个变种),javax提供了@Valid(标准JSR-303规范),配合BindingResult可以直接提供参数验证结果.其中对于字段的特定验证注解,比如@NotNull. @Valid和@Validated 批注 在Spring中,我们使用JSR

  • C#中的 == 和equals()区别浅析

    首先看看,如以下代码: int age = 25; short newAge = 25; Console.WriteLine(age == newAge); //true Console.WriteLine(newAge.Equals(age)); //false Console.ReadLine(); int和short为原始类型,但与"=="比较返回true,equals()比较返回false.为什么呢? 简而言之:"equals()"相比"= =&q

随机推荐