理解Java泛型 通配符 ? 以及其使用

什么是泛型:

泛型从字面上理解,是指一个类、接口或方法支持多种类型,使之广泛化、一般化和更加通用。Java中使用Object类来定义类型也 能实现泛型,但缺点是造成原类型信息的丢失,在使用中容易造成ClassCastException。

Java泛型带到的好处:

  1. 使得一个类或方法中的类型参数化,最终达到代码复用的效果。( 不使用泛型,你可能需要每种情况的类或方法都要定义一遍 )
  2. 实现类型检查的功能,避免ClassCastException。(这是相对于使用Object类型实现泛型而言。因为我可以每个类都定义一遍来实现所谓的类型检查。)

泛型自定义:类、接口、方法

//定义泛型类,接口的定义和类一样
class G1<T> {
    T content;
}

// 定义泛型方法,方法的头部使用<T>声明,注意结构
class GMethod1 {

    // 一般泛型方法定义
    public static <T> void method1(T params) {

    }

    // 返回值也为泛型
    public static <E> E method2(E params) {
        E content = params;
        return content;
    }

}

// extends 的使用,限定泛型的范围,等于或者是extends的子类;只有extends,没有super,通配符才有extends和super
class G2<T extends Number> {
    T content;
}

 

泛型的使用

    //泛型类或接口的使用
    List<String> list = new ArrayList<>();

    //泛型方法的使用,在方法前面使用<>传入类型
    <String>gMethod1.method1("str");
    <Integer>gMethod1.method1(new Integer(1));

    //如果入参的参数中使用了T,则可以省略方法前面的<>,编译器可以自行识别出其类型
    gMethod1.method1("str");
    gMethod1.method1(new Integer(1));

 

通配符?

什么时候会用通配符:

通配符只有在修饰一个变量时会用到,使用它可方便地引用包含了多种类型的泛型;

    public static void main() {

        //不使用通配符
        ArrayList<Object> arr = new ArrayList<Object>();
        // ArrayList<Object> arr = new ArrayList<String>(); 编译不通过,arr只能引用包含Object的集合

        //使用通配符
        ArrayList<?> arr2;
        arr2 = new ArrayList<String>();
        arr2 = new ArrayList<Integer>();
        arr2.get(0);    //返回的,是一个Object对象,通配符会使原集合包含类型信息丢失,也是通配符的使用代价

    // 通常在方法参数中才会使用通配符,使得这个方法可以引用多种泛型集合。这个和范型方法不一样,这里只是一个引用变量
    void gMethod(ArrayList<? extends Number> param) {
    }

 

可以看到,通配符使用方便的同时,使原集合包含类型信息丢失。

通配符的extends super关键字

详解:https://blog.csdn.net/qq_35923521/article/details/77717308

        ArrayList<? extends Number> arr3; // Number 是 Integer、Float的父类; ArrayList<Number> arr3只能引用 ArrayList<Number>
        arr3 = new ArrayList<Integer>();
        arr3 = new ArrayList<Float>();
        // arr3 = new ArrayList<String>(); 编译不通过,String 和 Number不存在继承关系

        arr3.get(0);    //返回的,是一个Number对象
        arr3.add(null); //使用过通配符修饰的集合变量,只能add(null),因为这个集合中包含的类型已经确定,只是类型信息已经丢失了,add(Object)也不行

 

无限定通配符?

ArrayList<?> arr3;  无通配符等同于 ArrayList<? extends Object> arr3;   //用于取值get(),不能赋值set()

 

扩展:不使用泛型的变量和另一种方式

        //这样使用功能和通配符一样,可以多种引用,但一般不推荐这样使用
        ArrayList a4;
        a4 = new ArrayList<String>();
        a4 = new ArrayList<Integer>();
        a4.add(new Integer(1));
        a4.add(new String("str"));//和通配符引用不能add不一样,这样方式可以add多种类型元素,但一般不推荐

        a4.get(0);  //返回的,是一个Object对象

 

posted @ 2017-07-25 00:36  mzzcy  阅读(10381)  评论(0编辑  收藏  举报