类型参数
泛型有三种实现方式,分别是泛型接口、泛型类、泛型方法,下面通过泛型方法来介绍什么是类型参数。
泛型方法声明方式:访问修饰符 <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 泛型中的通配符 T,E,K,V,?
原文:https://juejin.im/post/5d5789d26fb9a06ad0056bd9 前言 Java 泛型(generics)是 JDK 5 中引入的一个新特性, 泛型提供了编译时类型 ...
- 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 下界通 ...
- Java 泛型中的通配符
本文内容如下: 1. 什么是类型擦除 2.常用的 ?, T, E, K, V, N的含义 3.上界通配符 < ?extends E> 4.下界通配符 < ?super E> 5 ...
- Java泛型中的通配符
Java泛型中的通配符可以直接定义泛型类型的参数.而不用把该函数定义成泛型函数. public class GenericsTest { public static void main(String[ ...
- Java泛型中extends和super的理解(转)
E – Element (在集合中使用,因为集合中存放的是元素) T – Type(Java 类) K – Key(键) V – Value(值) N – Number(数值类型) ? – 表示不确定 ...
- Java泛型中extends和super的区别?
<? extends T>和<? super T>是Java泛型中的"通配符(Wildcards)"和"边界(Bounds)"的概念. ...
- Java泛型中的协变和逆变
Java泛型中的协变和逆变 一般我们看Java泛型好像是不支持协变或逆变的,比如前面提到的List<Object>和List<String>之间是不可变的.但当我们在Java泛 ...
- Java泛型中<?> 和 <? extends Object>的异同分析
相信很多人和我一样,接触Java多年,却仍旧搞不清楚 Java 泛型中 <?>和 <? extends Object>的相似和不同.但是,这应该是一个比较高端大气上档次的Que ...
- Java泛型中的细节
Java泛型中的细节 如果没有泛型 学习Java,必不可少的一个过程就是需要掌握泛型.泛型起源于JDK1.5,为什么我们要使用泛型呢?泛型可以使编译器知道一个对象的限定类型是什么,这样编译器就可以在一 ...
随机推荐
- 03.AOF持久化机制配置与工作流程
一.AOF持久化的配置 配置文件redis.conf,AOF持久化默认是关闭的,默认是打开RDB持久化 appendonly yes 二.工作流程: 打开AOF持久化机制之后,redis每次接 ...
- 【亲测】手把手教你如何破解pycharm(附安装包和破解文件)
此教程支持最新的2019.3版本的Pycharm,并兼容之前的版本. 一.准备工作: 1.下载Pycharm 有条件的可以自行去官网下载,这里我提供了我下载的版本,已上传到百度网盘,链接在下方. 2. ...
- unity3d屏幕截图功能
function OnGUI(){ if(GUI.Button(Rect(Screen.width*0.5-50,Screen.height*0.5-50,100,100),"screen& ...
- 记录一道有意思的js题目
偶然机会,在codewars上面开始做题,遇到一道有意思的题目,记录一下: 题目是这样的: In this kata, you will write a function that returns t ...
- 如何建立一个完美的 Python 项目
原文地址:How to set up a perfect Python project 原文作者:Brendan Maginnis 译者:HelloGitHub-丫丫 校对者:HelloGitHub- ...
- Java反射(一)
什么是反射? 在程序的运行过程中,可以动态的创建对象. 反射的基石是什么? 字节码对象是反射的基石.字节码对象:Java类文件通过javac进行编译后生成的xxx.class文件,此文件由jvm加载至 ...
- Codeforces Round #669 (Div. 2)A-C题解
A. Ahahahahahahahaha 题目:http://codeforces.com/contest/1407/problem/A 题解:最多进行n/2的操作次数,我们统计这n个数中1的个数,是 ...
- vue安装pubsub-js 库的命令
1.查看pubsub-js 库是否已经存在该库命令: npm info pubsub-js 2.若不存在,则先安装pubsub-js 库,命令如下: npm install --save pubsub ...
- leetcode刷题-91解码方法
题目 一条包含字母 A-Z 的消息通过以下方式进行了编码: 'A' -> 1'B' -> 2...'Z' -> 26给定一个只包含数字的非空字符串,请计算解码方法的总数. 示例 1: ...
- Linux通过命令行将英文改成中文
1.首先查看当前系统是否有中文语言包 locale -a 如果没有zh_CN.utf8,就需要下载中文语言包,否则,如果有中文语言包,跳过第二步. 2.安装中文语言包 Ubuntu: sudo apt ...