JAVA 泛型总结

什么是泛型

(1) 背景

JAVA推出泛型以前,程序员可以构建一个元素类型为Object的集合,该集合能够存储任意的数据类型对象,而在使用该集合的过程中,需要程序员明确知道存储每个元素的数据类型,否则很容易引发ClassCastException异常。

(2)泛型的概念

Java泛型(generics) 是JDK5中引入的一个新特性,泛型提供了编译时类型安全监测机制,该机制允许我们在编译时检测到非法的类型数据结构。泛型的本质就是参数化类型,也就是所操作的数据类型被指定为一个参数。

(3)泛型的好处

类型安全

消除了强制类型转换

2.泛型类 接口

泛型类

(1)泛型类的定义语法

// 常用的泛型标识: T、E、K、V class 类名称 <泛型标识,泛型标识,…>{ private 泛型标识 变量名; …… }

(2)泛型类的使用

使用语法

类名<具体的数据类型> 对象名 = new 类名<具体的数据类型>();

Java1.7以后,后面的<>中的具体的数据类型可以省略不写

类名<具体的数据类型> 对象名 = new 类名<>(); [菱形语法]

(3)注意事项

泛型类在创建对象的时候,来指定操作的具体数据类型,如果没有指定类型,将按照Object类型操作

泛型的类型参数不支持基本数据类型

同一泛型类,根据不同的数据类型创建的对象,本质上是同一类型

(4)从泛型类派生子类

子类也是泛型类,子类和父类的泛型类型要一致,子类可进行拓展使用其他泛型标识

class ChildGeneric extends Generic

class ChildGeneric<T,E> extends Generic

子类不是泛型类,父类要明确泛型的数据类型

class ChildGeneric extends Generic

泛型接口

(1)泛型接口的定义语法

interface 接口名称 <泛型标识,泛型标识,…> { 泛型标识 方法名(); }

(2)泛型接口的使用(和泛型类相似)

实现类不是泛型类,接口要明确数据类型

实现类也是泛型类,实现类的泛型标识要包含接口的泛型标识

3.泛型方法

(1)用法

泛型方法是在调用方法的时候指明泛型的具体类型

(2)语法

修饰符 <T,E,…> 返回值类型 方法名(形参列表) { 方法体… }

(3)说明

public与返回值中间非常重要,可以理解为声明此方法为泛型方法。

只有声明了的方法才是泛型方法,泛型类中的使用了泛型的成员方法并不是泛型方法。

< T >表明该方法将使用泛型类型T,此时才可以在方法中使用泛型类型T。

与泛型类的定义一样,此处T可以随便写为任意标识,常见的如T、E、K、V等形式的参数常用于表示泛型。

(4)泛型方法与可变参数

public void print(E… e){ for (E e1 : e) { System.out.println(e); } }

(5)泛型方法总结

泛型方法能使方法独立于类而产生变化

如果static方法要使用泛型能力,就必须使其成为泛型方法

4.类型通配符

(1)什么是类型通配符

类型通配符一般是使用"?"代替具体的类型实参。

类型通配符是类型实参,而不是类型形参。

(2)类型通配符的上限

语法:

类/接口<? extends 实参类型>

要求该泛型的类型,只能是实参类型,或实参类型的子类类型。

(3)类型通配符的下限

语法:

类/接口<? super 实参类型>

要求该泛型的类型,只能是实参类型,或实参类型的父类类型。

5.类型擦除

(1)概念

泛型是Java 1.5版本才引进的概念,在这之前是没有泛型的,但是泛型代码能够很好地和之前版本的代码兼容。那是因为,泛型信息只存在于代码编译阶段,在进入JVM之前,与泛型相关的信息会被擦除掉,我们称之为类型擦除。

无限制类型擦除

img

有限制类型擦除

img

擦除方法中类型定义的参数

img

桥接方法

img

6.泛型和数组

(1)泛型数组的创建

可以声明带泛型的数组引用,但是不能直接创建带泛型的数组对象

ArrayList[] listArr = new ArrayList<5>(); //会报错

//不会报错 ArrayList[] list = new ArrayList[5]; ArrayList[] listArr = list;

或者

ArrayList[] listArr = new ArrayList[5];

可以通过java.lang.reflect.Array的newInstance(Class,int)创建T[]数组

7.泛型和反射

反射常用的泛型类

Class Constructor  Class<Person> personClass = Person.class;  
Constructor<Person> constructor = personClass.getConstructor();
Person person = constructor.newInstance(); 

说说List,List,List<?>区别

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的父类型

下面举个栗子比较下这三种:

package com.lyang.demo.fanxing;

import java.util.Arrays;
import java.util.List;

/**
 * 测试泛型参数Object和T的区别  * Created by yanglu on 2017/04/20.
 */
public class TestDifferenceBetweenObjectAndT {
    public static void printList1(List<Object> list) {
        for (Object elem : list) System.out.println(elem + " ");
        System.out.println();
    }

    public static <T> void printList2(List<T> list) {
        for (T elem : list) System.out.println(elem + " ");
        System.out.println();
    }

    public static void printList3(List<?> list) {
        for (int i = 0; i < list.size(); i++) System.out.println(list.get(i) + " ");
        System.out.println();
    }

    public static void main(String[] args) {
        List<Integer> test1 = Arrays.asList(1, 2, 3);
        List<String> test2 = Arrays.asList("one", "two", "three");
        List<Object> test3 = Arrays.asList(1, "two", 1.23);
        List<Fruit> test4 = Arrays.asList(new Apple(), new Banana());        
        /*         * 下面这句会编译报错,因为参数不能转化成功         * */
        printList1(test4);         /**/
        printList1(test3);
        printList1(test3);
        printList2(test1);
        printList2(test2);
        printList2(test3);
        printList3(test1);
        printList3(test2);
        printList3(test3);
    }
} 

img

T,Class,Class<?>区别

T是一种具体的类,例如String,List,Map…等等,这些都是属于具体的类,这个比较好理解

Class是什么呢,Class也是一个类,但Class是存放上面String,List,Map…类信息的一个类**,有点抽象,我们一步一步来看 。

如何获取到Class类呢,有三种方式:

1. 调用Object类的getClass()方法来得到Class对象,这也是最常见的产生Class对象的方法。

例如:

List list = null; Class clazz = list.getClass();

2. 使用Class类的中静态forName()方法获得与字符串对应的Class对象。

例如:

Class clazz = Class.forName(“com.lyang.demo.fanxing.People”);

3.获取Class类型对象的第三个方法非常简单。如果T是一个Java类型,那么T.class就代表了匹配的类对象。

Class clazz = List.class;

那么问题来了?Class类是创建出来了,但是Class和Class<?>适用于什么时候呢???**

使用Class和Class<?>多发生在反射场景下,先看看如果我们不使用泛型,反射创建一个类是什么样的。

People people = (People) Class.forName(“com.lyang.demo.fanxing.People”).newInstance();

看到了么,需要强转,如果反射的类型不是People类,就会报

java.lang.ClassCastException错误。

使用Class泛型后,不用强转了

public class Test {
    public static <T> T createInstance(Class<T> clazz) throws IllegalAccessException, InstantiationException {
        return clazz.newInstance();
    }

    public static void main(String[] args) throws IllegalAccessException, InstantiationException {
        Fruit fruit = createInstance(Fruit.class);
        People people = createInstance(People.class);
    }
} 

那Class和Class<?>有什么区别呢?

Class在实例化的时候,T要替换成具体类

Class<?>它是个通配泛型,?可以代表任何类型,主要用于声明时的限制情况

例如可以声明一个

public Class<?> clazz;

但是你不能声明一个

public Class clazz;

因为T需要指定类型

所以当,不知道定声明什么类型的Class的时候可以定义一个Class<?>,Class<?>可以用于参数类型定义,方法返回值定义等。

参考

https://blog.csdn.net/slyzlh/article/details/116194892

https://github.com/qwertyanglu/FanxingDemo

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值