通过代码理解java泛型
泛型数据java基础,但真正理解需要悉心品尝。毕竟在工作中用到的是在是太多了。
不要以为newArrayList<>这就是泛型,这只能属于会使用。
在工作中,相对于现有的项目源码的数据库操作层,无论是mybatis,hibernate或者是自己封装的baseModel层,都会使用到泛型。
以及和这个屌东西。
泛型使用情况分为三类
1.泛型类。
2.泛型方法。
3.泛型接口。
出于规范的目的,Java还是建议我们用单个大写字母来代表类型参数。常见的如:
1.T代表一般的任何类。
2.E代表Element的意思,或者Exception异常的意思。
3.K代表Key的意思。
4.V代表Value的意思,通常与K一起配合使用。
5.S代表Subtype的意思,文章后面部分会讲解示意。
最直接的一段代码。
Listl1=newArrayList (); List l2=newArrayList (); System.out.println(l1.getClass()==l2.getClass());
打印的判断为TRUE,因为泛型信息被擦除了。
泛型擦除实例。
ListlistErasure=newArrayList (){ //直接初始化,这也是一种方式。直接传入一个collection。 {add("aaa");add("bbb");} }; listErasure.add("ccc"); Classclass1=listErasure.getClass(); Methodmethod=class1.getMethod("add",Object.class); method.invoke(listErasure,123); System.out.println(listErasure)
输出结果[aaa,bbb,ccc,123]
明明是接收String类型,但是却可以通过反射对其进行Integer类型的操作。
可见泛型只是在编译期间有效。
>代表着类型未知
和这个东西经常见到,但是并没有字面意义那么简单。
通配符有3种形式。
- >被称作无限定的通配符。
- 被称作有上限的通配符。
- 被称作有下限的通配符。
classA{} classBextendsA{} classCextendsB{}
ListlistExtends=newArrayList(); //listExtends.add(newA());全部编译错误。因为使用的是extends,丧失了写的操作能力。跟f3方法一样,是未知类型,只是确定了里面对象的范围。是B的子类。 //listExtends.add(newB()); //listExtends.add(newC()); //能进行对B以及B的子类操作。这是super的神奇之处。 ListlistSuper=newArrayList(); //listSuper.add(newA());//会编译错误。 listSuper.add(newB()); listSuper.add(newC());
以及方法泛型的返回
泛型作为参数的传递。
publicstaticTTTf1(TTTt){ returnt; } //传递指定的A类型,对应的list可以进行对应的list应有的方法。 publicstaticvoidf2(Listlist){ list.add(newA()); System.out.println(list.size()); } publicstaticvoidf3(List>list){ //list.add(newA());//当传入的是?通配符的话表示只能进行跟?无关的操作,类似于size方法,增加代码的可读性。 System.out.println(list.size()); } publicstaticvoidf4(ListlistExtends){ //listExtends.add(newB());//不能进行写做操,因为是?,增加了可读性。 System.out.println(listExtends.size()); }
测试代码,很全面
packagecom.javaSE.fanxing; importjava.lang.reflect.InvocationTargetException; importjava.lang.reflect.Method; importjava.util.ArrayList; importjava.util.List; classA{} classBextendsA{} classCextendsB{} publicclassDemo{ Tvalue; TTvalue2; publicTTgetValue2(){ returnvalue2; } publicvoidsetValue2(TTvalue2){ this.value2=value2; } publicTgetValue(){ returnvalue; } publicvoidsetValue(Tvalue){ this.value=value; } publicstatic TTTf1(TTTt){ returnt; } //传递指定的A类型,对应的list可以进行对应的list应有的方法。 publicstaticvoidf2(Listlist){ list.add(newA()); System.out.println(list.size()); } publicstaticvoidf3(List>list){ //list.add(newA());//当传入的是?通配符的话表示只能进行跟?无关的操作,类似于size方法,增加代码的可读性。 System.out.println(list.size()); } publicstaticvoidf4(ListlistExtends){ //listExtends.add(newB());//不能进行写做操,因为是?,增加了可读性。 System.out.println(listExtends.size()); } publicstaticvoidmain(String[]args)throwsNoSuchMethodException,SecurityException,IllegalAccessException,IllegalArgumentException,InvocationTargetException,InstantiationException{ //打印的判断为TRUE,因为泛型信息被擦除了。 List l1=newArrayList (); List l2=newArrayList (); System.out.println(l1.getClass()==l2.getClass()); //泛型擦除实例。 List listErasure=newArrayList (){ //直接初始化,这也是一种方式。直接传入一个collection。 {add("aaa");add("bbb");} }; listErasure.add("ccc"); Classclass1=listErasure.getClass(); Methodmethod=class1.getMethod("add",Object.class); method.invoke(listErasure,123); System.out.println(listErasure); Demo demo=newDemo (); demo.setValue("string"); System.out.println(demo.getValue()); Demo demo2=newDemo (); demo2.setValue(100); System.out.println(demo2.getValue()); System.out.println(f1(123)); //ListlistA=newArrayList(); //ListlistB=listA;//newArrayList();虽然B是A的子类,并不代表泛型之间也具备继承关系。 ArrayListlistA=newArrayList(); listA.add(newA()); f3(listA);//不对f3方法进行任何操作,是1. f2(listA);//2对应的方法实现还进行了一次插入操作。 f3(listA);//static,对应的listA的集合数量是引用值。 ArrayListlistB=newArrayList(); listB.add(newB()); f3(listB);//f3方法传递的是通配符?,不能进行add操作。 //和 ListlistExtends=newArrayList(); //listExtends.add(newA());全部编译错误。因为使用的是extends,丧失了写的操作能力。跟f3方法一样,是未知类型,只是确定了里面对象的范围。是B的子类。 //listExtends.add(newB()); //listExtends.add(newC()); //能进行对B以及B的子类操作。这是super的神奇之处。 ListlistSuper=newArrayList(); //listSuper.add(newA());//会编译错误。 listSuper.add(newB()); listSuper.add(newC()); //没毛病。 ListlistBS=newArrayList(); listBS.add(newB()); f4(listBS); } }
声明:本文内容来源于网络,版权归原作者所有,内容由互联网用户自发贡献自行上传,本网站不拥有所有权,未作人工编辑处理,也不承担相关法律责任。如果您发现有涉嫌版权的内容,欢迎发送邮件至:czq8825#qq.com(发邮件时,请将#更换为@)进行举报,并提供相关证据,一经查实,本站将立刻删除涉嫌侵权内容。