类型参数
泛型有三种实现方式,分别是泛型接口、泛型类、泛型方法,下面通过泛型方法来介绍什么是类型参数。
泛型方法声明方式:访问修饰符 <T,K,S...> 返回类型 方法名(方法参数){方法体}
一、访问修饰符与返回类型中间有个<T,K,S...>,T、K、S等属于类型参数,可以随便定义。
二、返回类型和方法参数可以是或者包含类型参数T、K、S等。
三、可以限定类型参数必须实现某些接口或者继承某个类,多个限定的类、接口中间用&分隔,类必须放在限定列表中所有接口的前面。
四、泛型方法可以定义在普通类中。
示例:
import java.io.Serializable;
 
public class Demo {
public static void main(String[] args) {
B b = new B();
C c = new C();
System.out.println(D.d(b, c));
}
}
 
class A {
@Override
public String toString() {
return "A{}";
}
}
 
class B extends A implements Serializable, Cloneable {
@Override
public String toString() {
return "B{}";
}
}
 
class C extends A {
@Override
public String toString() {
return "C{}";
}
}
 
class D {
public static <T extends A & Serializable & Cloneable, K> K d(T t, K k) {
System.out.println(t);
return k;
}
}
 
输出:
B{}
C{}
 
通配符类型
通配符类型可以理解为一种泛型调用时传递的一种特殊数据类型,表示允许类型参数在某个范围内变化。通配符类型有三种,分别是?、? extends、? super。它有什么用?创建如下三个类:
public class ParentClass{}
 
public class SonClass extends ParentClass{}
 
public class Operate<T> {
 
private T item;
 
public Operate(T item) {
this.item = item;
}
 
public T get() {
return item;
}
 
public void set(T item) {
this.item = item;
}
}
 
? extends X
子类型限定,表示泛型的类型参数不是固定的,而是X及其子类型。
如果存在Operate<ParentClass> operate1 = new Operate<SonClass>(new SonClass()),那么这行代码会报错,编译器提示类型不兼容,因为左边不是右边的基类。此时如果有这样一个方法:
public static void method(Operate<ParentClass> operate) {},
就无法将 new Operate<SonClass>(new SonClass()) 传递给这个方法。解决办法就是使用子类型限定定义方法参数:
public static void method(Operate<? extends ParentClass> operate) {},
就可以将 new Operate<SonClass>(new SonClass()) 传递给这个方法。
子类型限定的副作用是不能传递null以外的类型。
Operate<? extends ParentClass>的方法可以想象成下面这个样子(实际上不能这样写代码):
public ? extends ParentClass get() {
return item;
}
public void set(? extends ParentClass item) {
this.item = item;
}
此时get方法可以正常调用,因为返回的item肯定是ParentClass或者它的子类型。但是set方法就不能传递null以外的类型了,因为编译器只知道需要ParentClass或者它的子类型,但是不知道具体是哪个类,所以只能调用set(null)。如下:
public static void method(Operate<? extends ParentClass> operate) {
operate.get();
operate.set(null);
operate.set(new ParentClass());//报错
operate.set(new SonClass());//报错
}
 
? super X
超类型限定,表示泛型的类型参数不是固定的,而是X及其父类型。
Operate<? super SonClass>的方法可以想象成下面这个样子(实际上不能这样写代码):
public ? super SonClass get() {
return item;
}
public void set(? super SonClass item) {
this.item = item;
}
一、类型参数限定为X及其父类型,直至Object类,因为不知道具体是哪个父类型,因此方法返回的类型只能赋给Object。
二、只能传递null、X及其子类型,因为X及其子类型都是向上转型成X及其父类型。
Operate<? super SonClass> operate3 = new Operate<ParentClass>(new ParentClass())是成立的,所以可以将new Operate<ParentClass>(new ParentClass())传递给以下方法。
public static void method(Operate<? super SonClass> operate) {
SonClass sonClass = operate.get();//报错
ParentClass parentClass = operate.get();//报错
Object object = operate.get();
operate.set(new ParentClass());//报错
operate.set(null);
operate.set(new SonClass());
}
 
?
无类型限定,泛型的类型参数没有限定。
一、只能传递null类型。
二、方法返回的类型只能赋给Object。
new Operate<SonClass>(new SonClass())、new Operate<ParentClass>(new ParentClass())、new Operate<Object>(new Object())、new Operate<String>(new String())等都可以传递给以下方法。
public static void method(Operate<?> operate) {
SonClass sonClass = operate.get();//报错
ParentClass parentClass = operate.get();//报错
operate.set(new ParentClass());//报错
Object object = operate.get();
operate.set(null);
operate.set(new SonClass());//报错
}
有什么作用呢?对于一些不需要实际类型的方法,就显得比泛型方法可读性强,如下。
public static void method(Operate<?> operate) {
System.out.println(operate.get() == null);
}
public static <T> void method(Operate<T> operate) {
System.out.println(operate.get() == null);
}

Java泛型中的类型参数和通配符类型的更多相关文章

  1. 【转】聊一聊-JAVA 泛型中的通配符 T,E,K,V,?

    原文:https://juejin.im/post/5d5789d26fb9a06ad0056bd9 前言 Java 泛型(generics)是 JDK 5 中引入的一个新特性, 泛型提供了编译时类型 ...

  2. Java泛型中的通配符T,E,K,V

    Java泛型中的通配符T,E,K,V 1.泛型的好处 2.泛型中的通配符 2.1 T,E,K,V,? 2.2 ?无界通配符 2.3 上界通配符 < ? extends E> 2.4 下界通 ...

  3. Java 泛型中的通配符

    本文内容如下: 1. 什么是类型擦除 2.常用的 ?, T, E, K, V, N的含义 3.上界通配符 < ?extends E> 4.下界通配符 < ?super E> 5 ...

  4. Java泛型中的通配符

    Java泛型中的通配符可以直接定义泛型类型的参数.而不用把该函数定义成泛型函数. public class GenericsTest { public static void main(String[ ...

  5. Java泛型中extends和super的理解(转)

    E – Element (在集合中使用,因为集合中存放的是元素) T – Type(Java 类) K – Key(键) V – Value(值) N – Number(数值类型) ? – 表示不确定 ...

  6. Java泛型中extends和super的区别?

    <? extends T>和<? super T>是Java泛型中的"通配符(Wildcards)"和"边界(Bounds)"的概念. ...

  7. Java泛型中的协变和逆变

    Java泛型中的协变和逆变 一般我们看Java泛型好像是不支持协变或逆变的,比如前面提到的List<Object>和List<String>之间是不可变的.但当我们在Java泛 ...

  8. Java泛型中<?> 和 <? extends Object>的异同分析

    相信很多人和我一样,接触Java多年,却仍旧搞不清楚 Java 泛型中 <?>和 <? extends Object>的相似和不同.但是,这应该是一个比较高端大气上档次的Que ...

  9. Java泛型中的细节

    Java泛型中的细节 如果没有泛型 学习Java,必不可少的一个过程就是需要掌握泛型.泛型起源于JDK1.5,为什么我们要使用泛型呢?泛型可以使编译器知道一个对象的限定类型是什么,这样编译器就可以在一 ...

随机推荐

  1. 浅谈python垃圾回收机制

    引入 ​ 解释器在执行到定义变量的语法时,会申请内存空间来存放变量的值,而内存的容量是有限的,这就涉及到变量值所占用内存空间的回收问题,当一个变量值没有用了(简称垃圾)就应该将其占用的内存给回收掉,那 ...

  2. android开发之dip,dp与px像素之间的转换工具,可能用的不多,但是有总比没有好吧。

    作者:程序员小冰,CSDN博客:http://blog.csdn.net/qq_21376985,转载请说明出处. 下面是介绍: 免积分下载地址:http://download.csdn.net/de ...

  3. ios Standard Framework和Umbrella Framework

    Standard Framework:标准库,通过引用对应的header文件而不是引用master header 文件来引用类(也可以通过引用Master Header file来引用需要使用的类), ...

  4. Springboot-Mybatis-进阶

    目录 数据库关系 ResultMap association collection 动态sql if where set choose foreach 缓存 一级缓存 二级缓存 开启二级缓存 配置 异 ...

  5. 《神经网络的梯度推导与代码验证》之vanilla RNN的前向传播和反向梯度推导

    在本篇章,我们将专门针对vanilla RNN,也就是所谓的原始RNN这种网络结构进行前向传播介绍和反向梯度推导.更多相关内容请见<神经网络的梯度推导与代码验证>系列介绍. 注意: 本系列 ...

  6. POJ - 3851-Wormholes(SPFA判负环)

    A friend of yours, an inventor, has built a spaceship recently and wants to explore space with it. D ...

  7. Easy Problem(等差数列求和导公式)

    链接:https://ac.nowcoder.com/acm/contest/316/A 来源:牛客网 时间限制:C/C++ 1秒,其他语言2秒 空间限制:C/C++ 131072K,其他语言2621 ...

  8. 关于移动端的文本框获取焦点时导致fixed或absolute定位的按钮被手机键盘顶上去的问题

    var win_h = $(window).height();//关键代码 window.addEventListener('resize', function () { if($(window).h ...

  9. linux 文件类型和权限

    linux 文件类型和权限 ls -l 显示: [user@wyf-201 ~]$ ll total 0 -rw-rw-r--. 1 user user 0 Aug 27 10:49 1.txt dr ...

  10. 跟着尚硅谷系统学习Docker-【day07】

    day07-20200722   p27.dockerfile案例编写-3   第一步:编写父dockerfile DockerFile_20200722_2 FROM centos RUN yum ...