类型参数
泛型有三种实现方式,分别是泛型接口、泛型类、泛型方法,下面通过泛型方法来介绍什么是类型参数。
泛型方法声明方式:访问修饰符 <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. Java高级特性——反射机制(完结)——反射与注解

    按照我们的学习进度,在前边我们讲过什么是注解以及注解如何定义,如果忘了,可以先回顾一下https://www.cnblogs.com/hgqin/p/13462051.html. 在学习反射和注解前, ...

  2. JavaScript中pipe实战

    JavaScript中pipe原理 代码示例 const pipe = (...fns) => x => fns.reduce((y, f) => f(y), x); 原理 一行代码 ...

  3. 23种设计模式 - 组件协作(TemplateMethod - Observer/Event - Strategy)

    其他设计模式 23种设计模式(C++) 每一种都有对应理解的相关代码示例 → Git原码 ⌨ 组件协作 现代软件专业分工之后的第一个结果是"框架与应用程序的划分","组件 ...

  4. go语言之抛出异常

    一: panic和recover 作用:panic 用来主动抛出错误: recover 用来捕获 panic 抛出的错误. 概述: ,引发panic有两种情况 )程序主动调用panic函数 )程序产生 ...

  5. vue 在模板template中变量和字符串拼接

    例子:  :post-action="'/api/v1/reportPage/'+this.selectedPagerId+'/saveimg/'"

  6. js中call,apply和bind

    1,首先先做一个定义:每个函数都包含两个非继承的方法:apply()和call(),apply和call这两个方法的用途都是在特定的作用域中调用函数,实际上等于设置函数体内this对象的值,两者唯一的 ...

  7. 用python爬虫写一个属于自己的彩虹屁生成器!

    效果图如下:

  8. Java Android程序员软件开发知识:枚举的介绍,以及代码的编写教程。

    Java枚举 Java 5.0引入了枚举,枚举限制变量只能是预先设定好的值.使用枚举可以减少代码中的bug. 例如,我们为果汁店设计一个程序,它将限制果汁为小杯.中杯.大杯.这就意味着它不允许顾客点除 ...

  9. Android开发之java代码中获取当前系统的时间工具类

    /** * 获取当前时间 * * @return */ public String getTime() { Date date = new Date();// 创建一个时间对象,获取到当前的时间 Si ...

  10. 挂载磁盘不成功显示mount: /mnt: wrong fs type, bad option, bad superblock..............

    [23:25:32 root@8 ~]#mount /dev/sdb2 /mntmount: /mnt: wrong fs type, bad option, bad superblock on /d ...