泛型总结

来源:互联网 发布:阿玛尼没有入住淘宝吗 编辑:程序博客网 时间:2024/06/17 06:40

泛型最大的好处就是类型检查,尤其是对集合非常有用,另外在底层代码设计中很有用处,它实现了重用的功能。泛型有两种定义方式,一个是泛型类,另一个是泛型方法。

java
//泛型类
public class Person<T> {
private T t;
public T getT() {
return t;
}
public void setT(T t) {
this.t = t;
}
}

T就是类型变量,是一个参数化类型,用尖括号(<>)括起来,放在类名的后面。泛型类的类型变量可以定义多个。类型变量一般都使用一个大写字母表示,比如本例的Person,JDK中的List,Map

//泛型方法public <T> T get(String key, Object params) {   return (T) getSqlSession().selectOne(key, params);}

这是我在SSM3示例的MyBatisDao这个类里面定义的一个方法,此方法就是一个泛型方法。就是类型变量,而get前面的T是返回类型。其实这个方法是存在类型安全问题的,如果我在RoleService里面调用这个方法,将返回类型T写成User,编译器是不会有任何警告信息的。

但如果我改写一下,将MyBatisDao加上泛型,public class MyBatisDao extends SqlSessionDaoSupport

这时User的返回类型就出现编译错误了:

编译器根据MyBatisDao这个Role类型变量就会推断出它里面定义的get方法应该返回Role类型。不过这样改过之后MyBatisDao就变成泛型类了,而get方法也不再是泛型方法。那么泛型方法能不能有安全检查呢?有,但是需要一些编程技巧,关键还是跟你写的泛型方法有关系,后面提到的类型参数的限定可以对泛型加以约束,解决一些安全检查的问题。

类型参数的限定

对于像这样的类型变量所代表的范围有时太大了点,有时不方便使用。比如现在需要实现了java.io.Serializable接口的泛型类,那么这应该如何做呢?JDK那帮专家们为我们设计了一种叫做“有限制的通配符类型”来解决这个问题。一般我们称为上限 和下限,他们一般写成下面这样:

上限:或者

public void add(List<? extends T> list){    for(T t : list){       add(t);   }}public void add(T t){};public void add(T t, List<? super T> list){    list.add(t);}

泛型的擦除

泛型主要是在编译期有效,即编译的时候检查类型安全,现在写代码一般都会用Eclipse或IntelliJ,这些集成开发工具都能做到即时编译,哪里有错马上会出现红色的错误标识。因此如果出现类型转换错误,会很明显的看到结果。但是,在程序的运行阶段,JVM是不认识泛型是神马东西的,所有有泛型的类,接口,方法都会被擦除掉泛型,变成原生类型(raw type),即Person变为Person,List

public class com.bolo.Person extends java.lang.Object{   private java.lang.Object t;   public com.bolo.Person();   public java.lang.Object getT();   public void setT(java.lang.Object);   public void add(java.util.List);   public void add(java.lang.Object);   public void add(java.lang.Object, java.util.List);}

因此针对泛型擦除这一特点,我们需要注意这样几点:

1、JVM里面没有泛型,只有普通的类和方法。2、所有的类型参数都用限定类型或者无限定类型用Object来替代。3、请谨慎处理方法重载,错误的使用重载不会实现想要的多态。4、为保证类型安全,必要时请使用强制类型转换。

泛型的限制

基本类型不能作为类型参数。Person就是错误的,只能用Person

类型检查只能用原始类型。if(t instanceof Person) 如果写成if(t instanceof Person)马上会出现编译错误

不能实例化类型变量。这样写是错误的:T t = new T()

不能实例化参数化类型的数组。Person[] p = new Person[5] //ERROR

不能定义静态实例变量和静态方法。如果你想这样写:private static T a 那对不起,编译器马上会给你一个错误提示。

其实关于泛型的限制完全可以不用讲,现在编译器都很强大,只要你这样做了,马上会给你显示一个错误。

最后说下泛型对于集合的用处来说是最大的,集合是一个容器,有了泛型就更方便重用。而我们使用最频繁的集合就是List列表,还有一个容器就是数组,大象在这里强烈建议大家多用List,尽量或最好不要用数组。其一是List有类型安全性检查,其二是数组的功能List都提供了并且更丰富,其三List对gc进行了优化。如果使用数组,特别是操作对象数组,如果经验不足,没有释放数组里面的对象引用,则很容易造成内存泄漏的问题。

一般使用来声明类型持有者名称,自定义泛型类时,类持有者名称可以使用T(Type)
如果是容器的元素可以使用E(Element),若键值匹配可以用K(Key)和V(Value)等,
若是

例子:Set<T> 表示 集合里 是   T类的实例List<E> 表示  集合里 是  E类的实例List<?> 表示 集合里的对象类型不确定,未指定ListList<?> 是一样的。
原创粉丝点击