通过代码理解java泛型

 更新时间:2018年08月13日 08:46:07   作者:陈灬大灬海  
本篇文章通过代码实例让大家充分的理解java泛型的相关知识点内容,有需要的朋友们可以学习下。

泛型数据java基础,但真正理解需要悉心品尝。毕竟在工作中用到的是在是太多了。

不要以为new ArrayList<>这就是泛型,这只能属于会使用。

在工作中,相对于现有的项目源码的数据库操作层,无论是mybatis,hibernate或者是自己封装的baseModel层,都会使用到泛型。

以及<? extends T> 和 <? super T>这个屌东西。

泛型使用情况分为三类

1. 泛型类。

2. 泛型方法。

3. 泛型接口。

出于规范的目的,Java 还是建议我们用单个大写字母来代表类型参数。常见的如:

1. T 代表一般的任何类。

2. E 代表 Element 的意思,或者 Exception 异常的意思。

3. K 代表 Key 的意思。

4. V 代表 Value 的意思,通常与 K 一起配合使用。

5. S 代表 Subtype 的意思,文章后面部分会讲解示意。

最直接的一段代码。

List<String> l1 = new ArrayList<String>();
  List<Integer> l2 = new ArrayList<Integer>();
  System.out.println(l1.getClass() == l2.getClass());

打印的判断为TRUE,因为泛型信息被擦除了。

泛型擦除实例。

List<String> listErasure = new ArrayList<String>() {
   // 直接初始化,这也是一种方式。直接传入一个collection。
   {add("aaa");add("bbb");}
  };
  listErasure.add("ccc");
  Class<? extends List> class1 = listErasure.getClass();
  Method method = class1.getMethod("add",Object.class);
  method.invoke(listErasure, 123);
  System.out.println(listErasure)

输出结果  [aaa, bbb, ccc, 123]

明明是接收String类型,但是却可以通过反射对其进行Integer类型的操作。

可见泛型只是在编译期间有效。

<?> 代表着类型未知

<? extends T> 和 <? super T>这个东西经常见到,但是并没有字面意义那么简单。

通配符有 3 种形式。

  1. <?> 被称作无限定的通配符。
  2. <? extends T> 被称作有上限的通配符。
  3. <? super T> 被称作有下限的通配符。
class A{}
class B extends A{}
class C extends B{}
List<? extends B> listExtends = new ArrayList<B>();
//  listExtends.add(new A()); 全部编译错误。因为使用的是extends,丧失了写的操作能力。跟f3方法一样,是未知类型,只是确定了里面对象的范围。是B的子类。
//  listExtends.add(new B());
//  listExtends.add(new C());
  
  // 能进行对B以及B的子类操作。这是super的神奇之处。
  List<? super B> listSuper = new ArrayList<B>();
//  listSuper.add(new A());//会编译错误。
  listSuper.add(new B());
  listSuper.add(new C());

以及方法泛型的返回

泛型作为参数的传递。

public static <TTT>TTT f1(TTT t) {
  return t;
 }
 
 // 传递指定的A类型,对应的list可以进行对应的list应有的方法。
 public static void f2(List<A> list) {
  list.add(new A());
  System.out.println(list.size());
 }
 
 public static void f3(List<?> list) {
//  list.add(new A()); //当传入的是?通配符的话表示只能进行跟?无关的操作,类似于size方法,增加代码的可读性。
  System.out.println(list.size());
 }
 
 public static void f4(List<? extends B> listExtends) {
//  listExtends.add(new B());//不能进行写做操,因为是?,增加了可读性。
  System.out.println(listExtends.size());
 }

测试代码,很全面

package com.javaSE.fanxing;

import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.ArrayList;
import java.util.List;

class A{}
class B extends A{}
class C extends B{}
public class Demo<T,TT> {
 T value; 
 TT value2;
 public TT getValue2() {
  return value2;
 }
 public void setValue2(TT value2) {
  this.value2 = value2;
 }
 public T getValue() {
  return value;
 }
 public void setValue(T value) {
  this.value = value;
 }
 public static <TTT>TTT f1(TTT t) {
  return t;
 }
 // 传递指定的A类型,对应的list可以进行对应的list应有的方法。
 public static void f2(List<A> list) {
  list.add(new A());
  System.out.println(list.size());
 }
 public static void f3(List<?> list) {
//  list.add(new A()); //当传入的是?通配符的话表示只能进行跟?无关的操作,类似于size方法,增加代码的可读性。
  System.out.println(list.size());
 }
 public static void f4(List<? extends B> listExtends) {
//  listExtends.add(new B());//不能进行写做操,因为是?,增加了可读性。
  System.out.println(listExtends.size());
 }
 public static void main(String[] args) throws NoSuchMethodException, SecurityException, IllegalAccessException, IllegalArgumentException, InvocationTargetException, InstantiationException {
  // 打印的判断为TRUE,因为泛型信息被擦除了。
  List<String> l1 = new ArrayList<String>();
  List<Integer> l2 = new ArrayList<Integer>();
  System.out.println(l1.getClass() == l2.getClass());
  // 泛型擦除实例。
  List<String> listErasure = new ArrayList<String>() {
   // 直接初始化,这也是一种方式。直接传入一个collection。
   {add("aaa");add("bbb");}
  };
  listErasure.add("ccc");
  Class<? extends List> class1 = listErasure.getClass();
  Method method = class1.getMethod("add",Object.class);
  method.invoke(listErasure, 123);
  System.out.println(listErasure);
  Demo<String,Integer> demo = new Demo<String,Integer>();
  demo.setValue("string");
  System.out.println(demo.getValue());
  
  
  Demo<Integer,String> demo2 = new Demo<Integer,String>();
  demo2.setValue(100);
  System.out.println(demo2.getValue());
  
  
  System.out.println(f1(123));
//  List<A> listA = new ArrayList<A>();
//  List<B> listB = listA;//new ArrayList<B>();虽然B是A的子类,并不代表泛型之间也具备继承关系。
  
  
  ArrayList<A> listA = new ArrayList<A>();
  listA.add(new A());
  f3(listA); // 不对f3方法进行任何操作,是1.
  f2(listA); // 2对应的方法实现还进行了一次插入操作。
  f3(listA); // static ,对应的listA的集合数量是引用值。
  
  
  ArrayList<B> listB = new ArrayList<B>();
  listB.add(new B()); 
  f3(listB); // f3方法传递的是通配符?,不能进行add操作。
  
   
  // <? extends T> 和 <? super T>
  List<? extends B> listExtends = new ArrayList<B>();
//  listExtends.add(new A()); 全部编译错误。因为使用的是extends,丧失了写的操作能力。跟f3方法一样,是未知类型,只是确定了里面对象的范围。是B的子类。
//  listExtends.add(new B());
//  listExtends.add(new C());
  
  // 能进行对B以及B的子类操作。这是super的神奇之处。
  List<? super B> listSuper = new ArrayList<B>();
//  listSuper.add(new A());//会编译错误。
  listSuper.add(new B());
  listSuper.add(new C());
  
  
  // 没毛病。
  List<B> listBS = new ArrayList<B>(); 
  listBS.add(new B());
  f4(listBS); 
 }
}

相关文章

  • java中HashSet的特点及实例用法

    java中HashSet的特点及实例用法

    在本篇文章里小编给大家整理的是一篇关于java中HashSet的特点及实例用法,有兴趣的朋友们可以学习下。
    2021-04-04
  • Java 递归重难点分析详解与练习

    Java 递归重难点分析详解与练习

    一说起递归,我想每个人都不陌生。举个从小就听过的例子:从前有座山,山里有座庙,庙里有个和尚,和尚在讲故事,从前有座山,山里有座庙,庙里有个和尚,和尚在讲故事,从前有座山,要理解递归,就得先了解什么是递归,实际上这句话就是一个递归
    2021-11-11
  • javacv视频抽帧的实现过程详解(附代码)

    javacv视频抽帧的实现过程详解(附代码)

    这篇文章主要介绍了javacv视频抽帧的实现过程详解(附代码),视频抽帧可以做一些处理,比如水印,去水印等操作,然后再合成视频,需要的朋友可以参考下
    2019-07-07
  • java 实现增量同步和自定义同步的操作

    java 实现增量同步和自定义同步的操作

    这篇文章主要介绍了java 实现增量同步和自定义同步的操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2021-01-01
  • IDEA中创建maven项目webapp目录无法识别即未被标识的解决办法

    IDEA中创建maven项目webapp目录无法识别即未被标识的解决办法

    在学习SpringMVC课程中,基于IDEA新建maven项目模块后,webapp目录未被标识,即没有小蓝点的图标显示,所以本文给大家介绍了IDEA中创建maven项目webapp目录无法识别即未被标识的解决办法,需要的朋友可以参考下
    2024-03-03
  • 浅谈Mybatis+mysql 存储Date类型的坑

    浅谈Mybatis+mysql 存储Date类型的坑

    这篇文章主要介绍了浅谈Mybatis+mysql 存储Date类型的坑,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-11-11
  • JavaCV实现人脸检测功能

    JavaCV实现人脸检测功能

    这篇文章主要为大家详细介绍了JavaCV实现人脸检测功能,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-02-02
  • SpringBoot读取配置文件的五种方法总结

    SpringBoot读取配置文件的五种方法总结

    这篇文章主要为大家详细介绍了SpringBoot读取配置文件的五种方法,文中的示例代码讲解详细,对我们学习SpringBoot有一定帮助,需要的可以参考一下
    2022-08-08
  • SpringBoot返回对象时,如何将Long类型转换为String

    SpringBoot返回对象时,如何将Long类型转换为String

    这篇文章主要介绍了SpringBoot返回对象时,实现将Long类型转换为String,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-06-06
  • 线程池中使用spring aop事务增强

    线程池中使用spring aop事务增强

    这篇文章主要介绍了线程池中使用spring aop事务增强,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-02-02

最新评论