java中的泛型

本文将详细介绍java泛型的用法以及泛型的原理

 

java泛型

泛型是在J2 SE1.5中引入的一个特性 可以将类型抽象为一个参数 从而简化代码和实现类型安全

 

如何使用泛型

泛型可以用于方法 类和接口 通过尖括号加标识符的方式声明

class GenericityClass<T>
{
    T t;
    public GenericityClass(T t)
    {
        this.t = t;
    }
    public T getT()
    {
        return t;
    }
}

interface GenericityInterface<T>
{
    void printGenericity(T t);
}

public static <A> A test(A a)
{
    System.out.println(a);
    return a;
}

其中类型T是一个不确定的类型 需要填入实际类型才能使用

ps1: 可以给泛型参数起任意名称 例如 <T2333> <abcd> <FAN_XING>
ps2: 可以声明任意个泛型参数 不同的泛型参数用’,’隔开 例如 <T, E, TD250>

 

使用泛型类,方法与普通的类,方法并没有太大区别 除了要填入尖括号和类型

GenericityClass<String> gc = new GenericityClass<String>("泛型类测试");
String s = gc.getT();
Main.<String>test("泛型方法测试");

因为编译器推导的存在 我们可以省略一些重复的类型信息

GenericityClass<String> gc = new GenericityClass<>("泛型类测试");// 前边的<String>已经声明了类型 后边的就可以省略
String s = gc.getT();
Main.test("泛型方法测试");// 后边填入的参数String已经确定了泛型参数的类型 因此可以省略<String>

 

java泛型的实现原理

java的泛型实现机制并不是类似c++编译时模板生成 也不是c#运行时生成泛型类 而是利用了一套泛型擦除机制

通过观察编译后的字节码 我们发现 编译器将所有的泛型参数都处理成Object类型

class GenericityClass
{
    Object t;
    public GenericityClass(Object t)
    {
        this.t = t;
    }
    public Object getT()
    {
        return t;
    }
}

interface GenericityInterface
{
    void printGenericity(Object t);
}

public static Object test(Object a)
{
    System.out.println(a);
    return a;
}
GenericityClass gc = new GenericityClass("泛型类测试");
String s = (String) gc.getT();// 因为返回值是Object所以要强制转换
BlueBridge.test("泛型方法测试");

 

我们可以用反射来验证这一点  如果是泛型擦除 那么List<String>的实际类型应该是List<Object>

List<String> list = new ArrayList<>();
list.add("这是String");
list.add("这也是String");
list.add("这还是String");

try {
    List.class.getMethod("add", Object.class).invoke(list, (Object) new int[] {1, 2, 3});
} catch (IllegalAccessException | InvocationTargetException | NoSuchMethodException e) {
    e.printStackTrace();
}

for (Object obj : list)
{
    System.out.println(obj);
}

运行结果

这是String
这也是String
这还是String
[I@4554617c

于是我们在一个List<String>中成功插入了一个int数组

 

为什么java要使用泛型擦除机制来实现泛型

为了兼容旧版本代码

java的泛型是在J2 SE1.5中加入的 在此之前的通用容器都是将对象转为Object储存   List<T>编译后就成了List 完美兼容旧版本

但是这也带来了一些缺陷 泛型不能是基本类型 所有基本类型必须转为对应的包装类 例如List<Integer>  这个过程中产生了装箱 导致效率损失

听说之后版本的java会引入一些机制来解决基本类型装箱问题 但那可能得等到猴年马月了

 

END

    原文作者:redstone_torch
    原文地址: https://www.cnblogs.com/redstone-torch/p/11844553.html
    本文转自网络文章,转载此文章仅为分享知识,如有侵权,请联系博主进行删除。
点赞