-
泛化:可以用T代表任意类型,所以许多重要的类,比如集合框架,都已经成为泛型化的了,这带来了很多好处。
-
类型安全:使用泛型可以使编译器知道变量的类型限制,进而可以在更高程度上验证类型假设。如果不用泛型,则必须使用强制类型转换,而强制类型转换不安全,在运行期可能发生ClassCast Exception异常,如果使用泛型,则会在编译期就能发现该错误。
-
消除强制类型转换:泛型可以消除源代码中的许多强制类型转换,这样可以使代码更加可读,并减少出错的机会。
在Android源码当中有很多地方用到了泛型。
二、使用 2.1 Java泛型接口把泛型定义在接口,如下:
public interface 接口名<泛型类型> {
}
2.1.1 定义
使用场景:网络请求后调用接口传入某个实体类(未知),请求成功后返回该实例。如下:
public interface HttpResponse2.1.2 使用{ //请求成功 void onSuccess(T bean); //请求失败 void onError(String response); }
//HomeBean.class new HttpResponse() { ... }; //BannerBean.class new HttpResponse () { .... };
这个T可以是HomeBean 也可以是BannerBean。
2.2 Java泛型类把泛型定义在类上,如下:
public class 类名<泛型类型> {
}
2.2.1 定义
使用场景:我们用到的地方就更多了。如网络请求返回的data(经常被定义为泛型),如下:
public class ResponseData2.2.2使用{ private int errorCode; private String errorMsg; private T data; }
@GET("banner/json")
Call>> homeBannerRetrofit();
@POST("user/register")
@FormUrlEncoded
Call> registerRetrofit(@FieldMap Map map);
这个T可以是List
把泛型定义在类上,如下:
public <泛型类型> 返回类型 方法名<泛型类型 变量名> {
}
2.3.1 定义
使用场景:我们用到的地方就更多了。如网络请求返回的data(经常被定义为泛型),如下:
public class Test {
public T name(T data){
return data;
};
}
2.3.2 使用
Test test = new Test(); HomeBean homeBean = test.name(new HomeBean()); RegisterData registerData = test.name(new RegisterData());
这个T可以是HomeBean 也可以是RegisterData。
2.4 Java泛型擦除及其相关内容在编译期间,所有泛型信息都会被擦除掉,在生成的字节码中是不包括泛型中的类型信息的。
2.4.1 ArrayList源码public class ArrayListextends AbstractList implements List , RandomAccess, Cloneable, java.io.Serializable { }
这明显就是个泛型类。下面咱们看一组实例:
Listlist = new ArrayList<>(); list.add("abc"); List list1 = new ArrayList<>(); list1.add(123); List list2 = new ArrayList<>(); list2.add(new UserBean(20,"sc")); MLog.e(String.valueOf(list.getClass())); MLog.e(String.valueOf(list.getClass() == list1.getClass())); MLog.e(String.valueOf(list.getClass() == list2.getClass())); MLog.e(String.valueOf(list1.getClass() == list2.getClass()));
打印结果:
E/---mlog----: class java.util.ArrayList E/---mlog----: true E/---mlog----: true E/---mlog----: true
然后你会发现ArrayList
约定俗成的东西:
-
T (type) 表示具体的一个java类型
-
K V (key value) 分别代表java键值中的Key Value
-
E (element) 代表Element
也可以定义为其他字母,但是不推荐,比较你用这几个别人一看就知道什么意思。
2.5.2 extends T> 上界通配符上界通配符: extends T> 表示的是类型的上限就是自身,因此通配的参数化类型可能是T或T的子类。
代码如下:
private void test(){
// List extends YeYe> listZuZong = new ArrayList();//报错
List extends YeYe> listYeYe = new ArrayList();
List extends YeYe> listBaBa = new ArrayList();
List extends YeYe> listSuSu = new ArrayList();
List extends YeYe> listZiji = new ArrayList();
};
class ZuZong{
}
class YeYe extends ZuZong{
}
class BaBa extends YeYe{
}
class SuSu extends YeYe{
}
class Ziji extends BaBa{
}
2.5.3 super T> 上界通配符
下界通配符: super T> 表示的是类型的下限就是自身,因此通配的参数化类型可能是T或T的父类,一直朝上直到Object。
代码如下:
List super YeYe> listObject = new ArrayList2.5.4 > 无界通配符
无界通配符:任意类型。
代码如下:
List> listObject = new ArrayList



