Java 泛型

01.泛型

泛型的本质是参数化类型,使用泛型可以获得更高级的抽象。

Java泛型(generics)是JDK 5 引入的一个新特性,泛型提供了编译时类型安全检测机制,该机制允许程序员在编译时检测到非法的类型。

02.泛型方法

<li> 场景一:

假定我们有这样一个需求:写一个排序方法,能够对整型数组、字符串数组甚至其他任何类型的数组进行排序,该如何实现?


    /**

     * 这个方法可以接受任意类型(基础类型)的数组

     *

     * @param array

     * @param <T>   数组类型

     */

    private static <T> void printArray(T[] array) {

        if (array == null || array.length <= 0) {

            return;

        }

        for (T t : array) {

            System.out.println(String.valueOf(t));

        }

    }

使用泛型可以接受不同类型的参数。

<li> 场景二:

  • 实现一个方法,可以调用动物的run()函数。【这里有Person、Lion分别继承了Animal类,Animal类有run()方法】。

    /**

     * 泛型的类型还可以使用extends关键字来限制类型的边界

     * @param animal

     * @param <T>

     */

    private static <T extends Animal> void printRun(T animal) {

        if (animal == null) {

            return;

        }

        animal.run();

    }

要声明一个有界的类型参数,首先列出类型参数的名称,后跟extends关键字,最后紧跟它的上界。

03.泛型类

<li> 设计一个类,实现缓存、打印的功能,但是不限制传递的类型。



class Print<T> {



    private ArrayList<T> list = new ArrayList<>();



    void addItem(T t) {

        list.add(t);

    }



    void exec() {

        for (T t : list) {

            System.out.println(t.toString());

        }

    }

}

上边这个类完成了缓存、打印的功能,并且可以传递任意类型。增加了类的适用范围,是更高级的抽象。下面是使用方法:·


Print<String> print1 = new Print<String>();

print1.addItem("中华人民共和国");

print1.addItem("美利坚合众国");

print1.exec();



Print<Integer> print2 = new Print<Integer>();

print2.addItem(123);

print2.addItem(987);

print2.exec();

04.泛型接口



public interface Iterator<E> {

    

    boolean hasNext();



    E next();



    

   default void remove() {

        throw new UnsupportedOperationException("remove");

    }



   

    default void forEachRemaining(Consumer<? super E> action) {

        Objects.requireNonNull(action);

        while (hasNext())

            action.accept(next());

    }

}

Iterator接口是Java集合框架中非常重要的一个类,它规定了迭代的规则。那么为什么一定要使用泛型呢?因为它同时限制了next()方法的返回类型。在很多情况下使用泛型会使编程变得更容易。

05.类型通配符


   /**

     * 不限制类型

     * @param list

     */

    private static void printList(ArrayList<?> list) {

        if (list != null) {

            System.out.println(list.toString());

        }

    }

调用方法:


ArrayList<String> list1= new ArrayList<String>();

list1.add("中华人民共和国");

printList(list1);



ArrayList<Integer> list2= new ArrayList<Integer>();

list2.add(12232323);

printList(list2);

06. <? extends T>和<? super T>的区别

  • <? extends T>表示该通配符所代表的类型是T类型的子类。

  • <? super T>表示该通配符所代表的类型是T类型的父类。

07.Java中泛型是类型擦除的

Java 泛型(Generic)的引入加强了参数类型的安全性,减少了类型的转换,但有一点需要注意:Java 的泛型在编译器有效,在运行期被删除,也就是说所有泛型参数类型在编译后都会被清除掉,看下面一个列子,代码如下:


public class Foo {  

    public void listMethod(List<String> stringList){  

    }  

    public void listMethod(List<Integer> intList) {  

    }  

}  

代码很简单,看起来没什么问题,但是编译器却报出如下错误信息:

Method listMethod(List<String>) has the same erasure listMethod(List<E>) as another method in type Foo

此错误的意思是说它与另外一个方法重复,也就是方法签名重复。反编译之后的方法代码如下:

public void listMethod(List list)  {  }  

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 开发人员在使用泛型的时候,很容易根据自己的直觉而犯一些错误。比如一个方法如果接收List作为形式参数,那么如果尝试...
    时待吾阅读 1,078评论 0 3
  • 我们知道,使用变量之前要定义,定义一个变量时必须要指明它的数据类型,什么样的数据类型赋给什么样的值。 假如我们现在...
    今晚打肉山阅读 1,031评论 0 1
  • 为什么需要泛型? 通过泛型可以定义类型安全的数据结构,而无须使用实际的数据类型(可扩展)。这能够显著提高性能并得到...
    一只好奇的茂阅读 1,285评论 2 39
  • [TOC] 深入理解 Java 泛型 概述 泛型的本质是参数化类型,通常用于输入参数、存储类型不确定的场景。相比于...
    albon阅读 5,398评论 0 7
  • 我有一位朋友,你第一眼看见他,就会觉得他很着急。这种着急不是表面上的匆匆忙忙,而是对成功的渴望。我们相识在一场会议...
    何泽坤Ken阅读 967评论 0 1