Java 获取泛型的类型实例详解

Java 获取泛型的类型实例详解

Java 泛型实际上有很多缺陷,比如不能直接获取泛型的类型,不能获取带泛型类等。

以下方式是不正确的:

①.获取带泛型的类的类型

Class lstUClazz = List<User>.class

②获取局部变量泛型的类型

List<User> listUser = new ArrayList<User>();
Type genType = listUser.getClass().getClass().getGenericSuperclass();

Class templatClazz = null;

if(ParameterizedType.class.isInstance(genType)
{
  //无法获取到User类,或者可能获取到错误的类型,如果有同名且不带包名的泛型存在
  ParameterizedType parameterizedType = (ParameterizedType) genType;
 templatClazz = (Class) parameterizedType.getActualTypeArguments()[0];
}

那么,如何才能获取到泛型的类型

①.必须具有真实类型的存在

②.泛型的类型是明确的如(List<User>是明确的,List<T>是不明确的)

满足以上两点,我们就可以获取泛型的类型了

1.通过继承方式,明确类型,然后获取泛型类

public abstract class JdbcDaoSupport<T> {

  protected JdbcDaoSupport() {

  }
   public Class getTempalteType()
  {
     Class<T> clazz = (Class<T>) ((ParameterizedType) getClass()
        .getGenericSuperclass()).getActualTypeArguments()[0];
     return clazz;
  }
}

public class UserDao extends JdbcDaoSupport<User> {
}

public class Test{
    public static void main(String[] args)
    {
      UserDao dao = new UserDao();
      Class clazz = dao.getTemplateType();
      System.out.println(clazz.getName()); //输出 xxx.xxx.User
    }
}

2.获取类属性的泛型类型

public class Test extends ClassA<String> {
  private List<String> list;
  private Map<String, Object> map; 

  /***
   * 获取List中的泛型
   */
  public static void testList() throws NoSuchFieldException, SecurityException {
    Type t = Test.class.getDeclaredField("list").getGenericType();
    if (ParameterizedType.class.isAssignableFrom(t.getClass())) {
      for (Type t1 : ((ParameterizedType) t).getActualTypeArguments()) {
        System.out.print(t1 + ",");
      }
      System.out.println();
    }
  } 

  /***
   * 获取Map中的泛型
   */
  public static void testMap() throws NoSuchFieldException, SecurityException {
    Type t = Test.class.getDeclaredField("map").getGenericType();
    if (ParameterizedType.class.isAssignableFrom(t.getClass())) {
      for (Type t1 : ((ParameterizedType) t).getActualTypeArguments()) {
        System.out.print(t1 + ",");
      }
      System.out.println();
    }
  } 

  public static void main(String args[]) throws Exception {
    System.out.println(">>>>>>>>>>>testList>>>>>>>>>>>");
    testList();
    System.out.println("<<<<<<<<<<<testList<<<<<<<<<<<\n");
    System.out.println(">>>>>>>>>>>testMap>>>>>>>>>>>");
    testMap();
    System.out.println("<<<<<<<<<<<testMap<<<<<<<<<<<\n");
    System.out.println(">>>>>>>>>>>testClassA>>>>>>>>>>>");
    new Test().testClassA();
    System.out.println("<<<<<<<<<<<testClassA<<<<<<<<<<<");
  } 

}

3.获取局部变量的泛型的类型

List<User> lst = new ArrayList<User>(){};
Type genType = listUser.getClass().getClass().getGenericSuperclass();

Class templatClazz = null;

if(ParameterizedType.class.isInstance(genType)
{
  ParameterizedType parameterizedType = (ParameterizedType) genType;
 templatClazz = (Class) parameterizedType.getActualTypeArguments()[0];
}

实际上,我们发现,能获取到泛型的类型实际上都是进行了“继承”。当然如果能熟练运用上述技巧,可以做很多事情,比如开源项目Gson中的TypeToken就是利用上述技巧,实现json与复杂类型的转换的。

很多情况下,Class被用来当作参数,我们其实可以将带泛型的类作为参数传入

我们一般为了方便,很少去特定定义一个类,因此,我们需要使用如下方式了。

Class clz = new ArrayList<User>(){}.getClass();

例子

class JsonToObjectUtil {

 public static <T> T jsonToObject(Class<T> clz,List<String> jsonList){

 Type genType = clz.getClass().getGenericSuperclass();
 Class templatClazz = null;

 if(ParameterizedType.class.isInstance(genType));
 {
  ParameterizedType parameterizedType = (ParameterizedType) genType;
  templatClazz = (Class) parameterizedType.getActualTypeArguments()[0];
 }
 List<Object> lst = new ArrayList<Object>();
 /*****/
 if(templatClazz!=null && jsonList!=null)
 {
  for (String str : jsonList) {

  Gson gson = new Gson();
  Object fromJson = gson.fromJson(str, templatClazz);
  lst.add(fromJson);
  }

 }
 /*****/
 return (T) lst;
 }

 public static void main(String[] args) {

 List<String> jsonList = null;
 //略去一部分填充jsonList的逻辑

 Class superClazz = new ArrayList<User>(){}.getClass();
 List jsonToObject = JsonToObjectUtil.jsonToObject(superClazz, jsonList);

 }

}

感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!

(0)

相关推荐

  • Java泛型类型通配符和C#对比分析

    c#的泛型没有类型通配符,原因是.net的泛型是CLR支持的泛型,而Java的JVM并不支持泛型,只是语法糖,在编译器编译的时候都转换成object类型 类型通配符在java中表示的是泛型类型的父类 public void test(List<Object> c) { for(int i = 0;i < c.size();i++) { System.out.println(c.get(i)); } } //创建一个List<String>对象 List<String&g

  • Java8中对泛型目标类型推断方法的改进

    一.简单理解泛型 泛型是Java SE 1.5的新特性,泛型的本质是参数化类型,也就是说所操作的数据类型被指定为一个参数.通俗点将就是"类型的变量".这种类型变量可以用在类.接口和方法的创建中. 理解Java泛型最简单的方法是把它看成一种便捷语法,能节省你某些Java类型转换(casting)上的操作: 复制代码 代码如下: List<Apple> box = new ArrayList<Apple>();box.add(new Apple());Apple a

  • Java 泛型总结(一):基本用法与类型擦除

    简介 Java 在 1.5 引入了泛型机制,泛型本质是参数化类型,也就是说变量的类型是一个参数,在使用时再指定为具体类型.泛型可以用于类.接口.方法,通过使用泛型可以使代码更简单.安全.然而 Java 中的泛型使用了类型擦除,所以只是伪泛型.这篇文章对泛型的使用以及存在的问题做个总结,主要参考自 <Java 编程思想>. 这个系列的另外两篇文章: Java 泛型总结(二):泛型与数组 Java 泛型总结(三):通配符的使用 基本用法 泛型类 如果有一个类 Holder 用于包装一个变量,这个变

  • java 用泛型参数类型构造数组详解及实例

    java 用泛型参数类型构造数组详解及实例 前言: 前一阵子打代码的时候突然想到一个问题.平时我们的数组都是作为一个参数传入方法中的,如果我们要想在方法中创建一个数组怎么样呢?在类型明确的情况下,这是没什么难度的.如果我们传入的参数是泛型类型的参数呢? public static <T> T[] creArray (T obj){ T[] arr = new T[10]; } 像上面这种用T来直接new数组的方法是错误的,会编译时出现一个:Cannot create a generic arr

  • JAVA利用泛型返回类型不同的对象方法

    有时需要在方法末尾返回类型不同的对象,而return 语句只能返回一个或一组类型一样的对象.此时就需要用到泛型. 首先先解释个概念, 元组:它是将一组对象直接打包存储于其中的一个单一对象,这个容器对象允许读取其中元素,但不能修改. 利用泛型创建元组 public class ReturnTwo<A,B> { public final A first; public final B second; public ReturnTwo(A a,B b) { first = a; second = b

  • Java泛型映射不同的值类型详解及实例代码

    Java泛型映射不同的值类型详解 前言: 一般来说,开发人员偶尔会遇到这样的情形: 在一个特定容器中映射任意类型的值.然而Java 集合API只提供了参数化的容器.这限制了类型安全地使用HashMap,如单一的值类型.但如果想混合苹果和梨,该怎样做呢? 幸运的是,有一个简单的设计模式允许使用Java泛型映射不同的值类型,Joshua Bloch在其<Effective Java>(第二版,第29项)中将其描述为类型安全的异构容器(typesafe hetereogeneous Containe

  • 详谈Java8新特性泛型的类型推导

    1. 泛型究竟是什么? 在讨论类型推导(type inference)之前,必须回顾一下什么是泛型(Generic).泛型是Java SE 1.5的新特性,泛型的本质是参数化类型,也就是说所操作的数据类型被指定为一个参数.通俗点将就是"类型的变量".这种类型变量可以用在类.接口和方法的创建中.理解Java泛型最简单的方法是把它看成一种便捷语法,能节省你某些Java类型转换(casting)上的操作: List<Apple> box = new ArrayList<Ap

  • 浅谈Java泛型让声明方法返回子类型的方法

    泛型典型的使用场景是集合.考虑到大多数情况下集合是同质的(同一类型),通过声明参数类型,可免去类型转换的麻烦.本文将讨论本人阅读Spring Security源码时遇到的一个关于泛型递归模式的问题. 声明方法返回子类型 在Spring Security的源码里有一个ProviderManagerBuilder接口,声明如下 public interface ProviderManagerBuilder<B extends ProviderManagerBuilder<B>> ext

  • Java 获取泛型的类型实例详解

    Java 获取泛型的类型实例详解 Java 泛型实际上有很多缺陷,比如不能直接获取泛型的类型,不能获取带泛型类等. 以下方式是不正确的: ①.获取带泛型的类的类型 Class lstUClazz = List<User>.class ②获取局部变量泛型的类型 List<User> listUser = new ArrayList<User>(); Type genType = listUser.getClass().getClass().getGenericSuperc

  • Java 获取服务器环境的实例详解

    Java 获取服务器环境的实例详解 废话不多说,直接上代码,代码中的注释写的比较清楚,请大家好好看, package com.rapido.utils; import java.util.Properties; /** * 获取服务器环境信息 * @author X-rapido * */ public class SystemDataUtil { public static Properties getSystemProperties() { Properties props = System

  • Java函数式接口Supplier接口实例详解

    这篇文章主要介绍了Java函数式接口Supplier接口实例详解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 JDK提供了大量常用的函数式接口以丰富Lambda的典型使用场景,它们主要在 java.util.function 包中被提供. 下面是最简单的Supplier接口及使用示例. Supplier接口概述 // Supplier接口源码 @FunctionalInterface public interface Supplier<T>

  • java操作mongoDB查询的实例详解

    java操作mongo查询的实例详解 前言: MongoDB是一个基于分布式文件存储的数据库.由C++语言编写.旨在为WEB应用提供可扩展的高性能数据存储解决方案. MongoDB是一个介于关系数据库和非关系数据库之间的产品,是非关系数据库当中功能最丰富,最像关系数据库的.他支持的数据结构非常松散,是类似json的bson格式,因此可以存储比较复杂的数据类型.Mongo最大的特点是他支持的查询语言非常强大,其语法有点类似于面向对象的查询语言,几乎可以实现类似关系数据库单表查询的绝大部分功能,而且

  • Java Web请求与响应实例详解

    Servlet最主要作用就是处理客户端请求并作出回应,为此,针对每次请求,Web容器在调用service()之前都会创建两个对象,分别是HttpServletRequest和HttpServletResponse.其中HttpServletRequest封装HTTP请求消息,HttpServletResponse封装HTTP响应消息.需要注意的是,Web服务器运行过程中,每个Servlet都会只创建一个实例对象,不过每次请求都会调用Servlet实例的service(ServletRequest

  • Java Exception 捕获和显示实例详解

    Java Exception 捕获和显示实例详解 在进行Java B/S架构开发时,经常有这样的场景:后端处理业务逻辑,前端负责展示,当后端处理出现异常时,如何把错误信息展示给前台呢?错误信息栈通常很多,对开发人员查找问题比较方便,但对于客户来说,打一堆的错误信息,无疑是对他们感官的一种摧残,如何捕捉最重要的信息显示到客户端呢?该信息要求简明扼要,指向出错点,且应指明异常的类型. 在很多情况下Exception的 getMessage()方法返回空的值,如果使用该方式则会在前端显示空值.我们要显

  • Java静态工厂方法的实例详解

     Java静态工厂方法的实例详解 什么是静态工厂方法 对于类而言,为了让使用者获取它自身的一个实例,最常用的方法就是提供一个公有的构造器. 当然,这里要介绍的是另一种方法--静态工厂方法,一个返回类的实例的静态方法. 举个例子,Boolean的一个将基本类型boolean转为封装类的方法,valueOf: public static Boolean valueOf(boolean b) { return (b ? TRUE : FALSE); } 为什么要使用静态工厂方法 那么,我们为什么要使用

  • 为Vue3 组件标注 TS 类型实例详解

    目录 为 props 标注类型 使用 <script setup> 非 <script setup> 为 emits 标注类型 使用 <script setup> 非 <script setup> 为 ref() 标注类型 默认推导类型 通过接口指定类型 通过泛型指定类型 为 reactive() 标注类型 默认推导类型 通过接口指定类型 为 computed() 标注类型 默认推导类型 通过泛型指定类型 为事件处理函数标注类型 为 provide / in

  • java 中自定义OutputFormat的实例详解

    java 中 自定义OutputFormat的实例详解 实例代码: package com.ccse.hadoop.outputformat; import java.io.IOException; import java.net.URI; import java.net.URISyntaxException; import java.util.StringTokenizer; import org.apache.hadoop.conf.Configuration; import org.apa

  • java 中createStatement()方法的实例详解

    java 中createStatement()方法的实例详解 用缺省设置创建时,ResultSet 是一种只能访问一次(one-time-through).只能向前访问(forward-only)和只读的对象.您只能访问数据一次,如果再次需要该 数据,必须重新查询数据库. 然而,并不只有这一种方式.通过设置 Statement 对象上的参数,您可以控制它产生的 ResultSet.例如: ... Class.forName(driverName); db = DriverManager.getC

随机推荐