Java点滴

Java 泛型

2017-11-02  本文已影响8人  一亩水塘

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> 场景二:


    /**

     * 泛型的类型还可以使用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>的区别

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)  {  }  

上一篇下一篇

猜你喜欢

热点阅读