Java 反射类型Type的用法说明

各个方法

1. 得到class的成员变量

首先得到object的class对象

然后在class对象中用getDeclaredFields()方法来获得class的成员变量

FieldTest ft = new FieldTest();
Class ftClass = ft.getClass();
Field[] fields = ftClass.getDeclaredFields();

2. field的函数

Field对象有很多成员方法

getName()获取名称。

getGenericType() 返回一个Type对象

getType() 返回Class对象

getGenericType 和getType区别:

返回类型一个是Class对象一个是Type接口。

如果属性是泛型,getType()返回属性的接口类型。getGenericType()还能返回参数类型。

String fieldName = field.getName();
Type genericType = field.getGenericType();
boolean isParameterizedType = (genericType instanceof ParameterizedType);
Class fieldClazz = field.getType();
String valueTypeName = genericType.getTypeName();

3.获取范型的Type和Class

获取 范型的key和value的Type

Type[] genericTypes = ((ParameterizedType) genericType).getActualTypeArguments();
Type type0 = genericTypes[0];
Type type1 = genericTypes[1];

通过google的com.google.common.reflect.TypeToken.of(type1).getRawType()方法获取map的key或者value的class类型。

Class<?> clazz = com.google.common.reflect.TypeToken.of(type1).getRawType();

总代码

class FieldTest {
    private String pri;
    protected String pro;
    public Map<Integer, HashMap<Integer, Float>> fcmap;
    public FieldTest() {
    }
    public FieldTest(String pri, String pro, String pub) {
        this.pri = pri;
        this.pro = pro;
    }
}
package cn.hyperchain.hvm.abi;
import java.lang.reflect.Array;
import java.lang.reflect.Field;
import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.util.Arrays;
import java.util.HashMap;
import java.util.Map;
public class Test {
    private static boolean checkClazzIsSpecific(Class CClazz, Class specific) {
        if (CClazz == specific) return true;
        Class[] interfaces = CClazz.getInterfaces();
        boolean result = false;
        for (Class inter : interfaces) {
            if (result) break;
            if (inter == specific) {
                result = true;
                break;
            }
            result = checkClazzIsSpecific(inter, specific);
        }
        return result;
    }
    public static void main(String args[]) {
        FieldTest ft = new FieldTest();
        Class ftClass = ft.getClass();
        Field[] fields = ftClass.getDeclaredFields();
        for (int i = 0; i < fields.length; i++) {
            Field field = fields[i];
            String fieldName = field.getName();
            Type genericType = field.getGenericType();
            Class fieldClass = field.getClass();
            boolean isParameterizedType = (genericType instanceof ParameterizedType);
            Class fieldClazz = field.getType();
            String valueTypeName = genericType.getTypeName();
            System.out.println();
            System.out.println("-------------------------------------------------");
            System.out.println();
            System.out.println("fieldName: " + fieldName);
            System.out.println("genericType: " + genericType);
                   System.out.println("fieldClazz: " + fieldClazz);
            System.out.println("isParameterizedType: " + isParameterizedType);
            System.out.println("valueTypeName: " + valueTypeName);
            if (checkClazzIsSpecific(fieldClazz, Map.class)) {
                Type[] genericTypes = ((ParameterizedType) genericType).getActualTypeArguments();
                Type type0 = genericTypes[0];
                Type type1 = genericTypes[1];
                String type1Name = genericTypes[1].getTypeName();
                System.out.println("type0: " + type0);
                System.out.println("type1: " + type1);
                System.out.println("type1Name: " + type1Name);
                System.out.println(type1 instanceof ParameterizedType);
                Type type3 = ((ParameterizedType)type1).getOwnerType();
                Class type4 = type1.getClass();
                //Class<?> type5 = (Class<?>) type1;
                System.out.println("type3: " + type3);
                System.out.println("type4: " + type4);
                //System.out.println(type5);
                Class<?> clazz = com.google.common.reflect.TypeToken.of(type1).getRawType();
                System.out.println("clazz: " + clazz);
            }
        }
    }
}
class FieldTest {
    private String pri;
    protected String pro;
//    public Map<String, Integer> map;
    public Map<Integer, HashMap<Integer, Float>> fcmap;
    public FieldTest() {
    }
    public FieldTest(String pri, String pro, String pub) {
        this.pri = pri;
        this.pro = pro;
    }
}
class abc {
    private String pri;
    protected String pro;
    public String pub;
    public String[] string;
    public int[] innt;
    public Map<String, Integer> map;
    public abc() {
    }
    public abc(String pri, String pro, String pub) {
        this.pri = pri;
        this.pro = pro;
        this.pub = pub;
    }
}

结果:

-------------------------------------------------

fieldName: pri
genericType: class java.lang.String
fieldClazz: class java.lang.String
isParameterizedType: false
valueTypeName: java.lang.String

-------------------------------------------------

fieldName: pro
genericType: class java.lang.String
fieldClazz: class java.lang.String
isParameterizedType: false
valueTypeName: java.lang.String

-------------------------------------------------

fieldName: fcmap
genericType: java.util.Map<java.lang.Integer, java.util.HashMap<java.lang.Integer, java.lang.Float>>
fieldClazz: interface java.util.Map
isParameterizedType: true
valueTypeName: java.util.Map<java.lang.Integer, java.util.HashMap<java.lang.Integer, java.lang.Float>>
type0: class java.lang.Integer
type1: java.util.HashMap<java.lang.Integer, java.lang.Float>
type1Name: java.util.HashMap<java.lang.Integer, java.lang.Float>
true
type3: null
type4: class sun.reflect.generics.reflectiveObjects.ParameterizedTypeImpl
clazz: class java.util.HashMap

Process finished with exit code 0

补充:Java-使用反射获取类型信息

Java中如何使用反射获取类型信息?

最近写了大量需要根据类属性的类型反射注入值的代码,总结了以下常用的反射技巧:

一个简单类的例子

在这个类中,有普通的String类型,有数组类型,有带泛型的List类型,有嵌套List类型,以及有多个泛型参数的简单类,这个类将作为我们后面的内容的基础。我们这一次博客解析如何使用反射获取到不同属性的类型值。

public class Some{
    private String name;
    private Integer[] numbers;
    private List<String> list;
    private List<List<Double>> matrix;
    private Map<String,Class> map;
    //ignore getter and setter
}

分析如何获取不同属性的类型

1、普通类型

普通类型的变量直接field.getType()即可以获取到他们的类型

public void queryNameType() throws NoSuchFieldException {
    Field field = Some.class.getDeclaredField("name");
    Class<?> type = field.getType();
    assertEquals(type,String.class);
}

2、数组类型

数组类型不像其他的类型可以通过isAssignableFrom()函数来进行判断,他需要使用isArray() 来判断该type是否是一个数组类型,然后使用getComponentType() 获取他的元素的类型

public void queryArrayType() throws NoSuchFieldException {
    Field field = Some.class.getDeclaredField("numbers");
    Class<?> type = field.getType();
    //一般来说,判断是否是某种类型是可以使用isAssignableFrom
    // 判断是否是数组类型比较特殊,要使用isArray()这个函数
    if (type.isArray()){
        //获得数组的类型,使用getComponentType()这个方法
        Class<?> componentType = type.getComponentType();
        assertEquals(componentType,Integer.class);
    }
    else{
        throw new IllegalStateException();
    }
}

3、带泛型的类型

带泛型的类型就是类似于List<String>这样的类型,我们现在的任务就是获取到String这个类型。

ParameterizedType表示参数化的类型,例如Collection这样的类型。我们可以通过getGenericType()方法获得该子类,当你的类型带有参数的时候就会返回ParameterizedType,否则会返回普通的类型(class)

那么具体是怎么操作的呢?

以获得List<T>的类型为例子

public void getListType() throws NoSuchFieldException {
    Field field = Some.class.getDeclaredField("list");
    //如果类似于List<String>这样的类型就是一种GenericType
    //注意这是一种Type类型
    Type type = field.getGenericType();
    if (type instanceof ParameterizedType){
        //泛型参数类型
        ParameterizedType parameterizedType = (ParameterizedType)type;
        Type[] actualTypes = parameterizedType.getActualTypeArguments();
        //因为List<String>获得第一个泛型参数,因为只有一个,我们取第一个
        //如果我们有多个泛型参数,我们可以根据顺序取不同的泛型参数
        assertEquals(actualTypes[0],String.class);
        //如果获得List这个原始类型呢?
        assertEquals(parameterizedType.getRawType(),List.class);
    }else{
        throw new IllegalStateException();
    }
}

4、复杂的嵌套类型

假如是List<List<String>> 如何获得最里面的类型呢?

例子如下

public void getSubListType() throws NoSuchFieldException {
  //思考一下,如果我们有一个嵌套List,我们想拿到嵌套在最里面的类型,那么我们可以这么做呢?
  //其实我们可以使用递归的思想去获得最里面的类型
  Field field = Some.class.getDeclaredField("matrix");
  assertEquals(getBaseType(field.getGenericType()),Double.class);
 }
 public static Type getBaseType(Type genericReturnType){
  Objects.requireNonNull(genericReturnType);
  if (genericReturnType instanceof ParameterizedType &&
    List.class.isAssignableFrom((Class)(((ParameterizedType) genericReturnType).getRawType()))){
   Type[] actualTypeArguments = ((ParameterizedType)genericReturnType).getActualTypeArguments();
   Type type = actualTypeArguments[0];
   return getBaseType(type);
  }else{
   return genericReturnType;
  }
 }

5、多个泛型参数

与第三个例子相似,只需要使用actualTypes数组按顺序取即可

例子如下

public void getMapType() throws NoSuchFieldException {
    Field field = Some.class.getDeclaredField("map");
    Type type = field.getGenericType();
    if (type instanceof ParameterizedType){
        ParameterizedType parameterizedType = (ParameterizedType)type;
        Type[] actualTypes = parameterizedType.getActualTypeArguments();
        assertEquals(actualTypes[0],String.class);
        assertEquals(actualTypes[1],Class.class);
    }else{
        throw new IllegalStateException();
    }
}

总结

以上总结了几种常用的使用反射获取属性类型的例子,稍加改造就可以写自己的工具类了。希望对大家有帮助^_^

(0)

相关推荐

  • java 通过反射遍历所有字段修改值的实例代码

    先给大家介绍下java遍历所有字段修改值的代码,具体内容详情如下所示: java 通过反射遍历所有字段修改值,避免重复set.get 比如一张表里的字段十几个,而这个表里的图片存储字段有八九个,在返回这个实体类的时候,要对图片进行加密或者其他操作,那就要在实体类查询结果出来后,一个个的get修复,再set赋值,代码量很多,另外如果有多个接口用到,就会产生重复代码: 通过java 的反射,遍历所有字段,进行一个判断,取出来的值是带有图片链接的,进行操作,省去了很多代码,下面贴代码 import o

  • java高级用法之注解和反射讲义

    前言 反射和注解在java中偏高级用法,一般在各种框架中被广泛应用,文章简单介绍下反射和注解的用法,希望对你的工作学习有一定帮助 java注解 什么是注解 Java 注解也就是Annotation是从 Java5 开始引入的新技术 Annotation的作用: 不是程序本身,可以对程序作出解释 可以被其他程序(编译器等)读取 Annotation的格式: 注解以@注释名在代码中存在的,可以添加一些数值,例如SuppressWarnings(value="unchecked") Anno

  • Java中反射机制和作用详解

    前言 很多刚学Java反射的同学可能对反射技术一头雾水,为什么要学习反射,学习反射有什么作用,不用反射,通过new也能创建用户对象. 那么接下来大师就带你们了解一下反射是什么,为什么要学习反射? 下面我们首先通过一个实例来说明反射的好处: 方法1.不用反射技术,创建用户对象,调用sayHello方法 1.1 我们首先创建一个User类 package com.dashi; /** * Author:Java大师 * User对象,包含用户的id和姓名以及sayHello方法 */ public

  • 如何用Java注解和反射实现依赖注入

    概述 在Spring中,我们可以通过 @Autowired注解的方式为一个方法中注入参数,那么这种方法背后到底发生了什么呢,这篇文章将讲述如何用Java的注解和反射实现一个"低配版"的依赖注入. 下面是我们要做的一些事情: 通过 @interface的方式定义一个注解 为某个希望杯被注入的方法添加这个注解 编写测试代码,通过反射获取添加了注解的方法对应的Method对象,将该方法对象设置为可访问的,通过反射创建对象并调用这个方法,同时注入依赖数据 如上所述,我们分为三个步骤, 去加工出

  • Java进阶知识之反射的概念与获取方法

    一.反射的基本概念 1.反射的概念 将类的各个组成部分封装成其他对象,这就是反射的机制. 通过上图,我们可以通过反射获得Person类的Class对象,通过获得Class对象来获得其中的成员变量,构造方法和成员方法,这个也是我们使用反射的目的. 2.使用反射的好处 可以在程序的运行过程中操作这些对象,获得类对象的属性,方法等. 可以解耦,以此来提高程序的可扩展性 3.补充 反射:框架设计的灵魂,反射是框架设计的关键点 那么知道了反射,框架又是什么呢 框架:半成品软件,我们在开发的过程中可以在框架

  • java 非常好用的反射框架Reflections介绍

    Reflections通过扫描classpath,索引元数据,并且允许在运行时查询这些元数据. 使用Reflections可以很轻松的获取以下元数据信息: 1)获取某个类型的所有子类:比如,有一个父类是TestInterface,可以获取到TestInterface的所有子类. 2)获取某个注解的所有类型/字段变量,支持注解参数匹配. 3)使用正则表达式获取所有匹配的资源文件 4)获取特定签名方法. 通常的用法有: 引入依赖jar <dependency> <groupId>org

  • Java基础之反射详解

    前言 反射是我们框架的灵魂,反射也是我们框架的一个底层基石,没有反射也就没有框架,如果我们学好了反射,对我们阅读框架底层是有很大班助的--阿俊.有些文章上来就讲反射,就会很懵逼,不知道是干啥的,所以我们就引出一些问题来看看为什么需要反射 一.一个需求引出反射 看下面的问题 根据配置文件reflection.properties指定信息,创建People对象并调用方法hi classullpath= com.reflection.People method=hi 思考:使用现有技术,能做吗? 我们

  • Java 反射类型Type的用法说明

    各个方法 1. 得到class的成员变量 首先得到object的class对象 然后在class对象中用getDeclaredFields()方法来获得class的成员变量 FieldTest ft = new FieldTest(); Class ftClass = ft.getClass(); Field[] fields = ftClass.getDeclaredFields(); 2. field的函数 Field对象有很多成员方法 getName()获取名称. getGenericTy

  • Java 反射机制原理与用法详解

    本文实例讲述了Java 反射机制原理与用法.分享给大家供大家参考,具体如下: 反射反射,程序员的快乐! 1.什么是反射? Java反射就是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法:对于任意一个对象,都能够调用它的任意方法和属性:并且能改变它的属性.而这也是Java被视为动态(或准动态,为啥要说是准动态,因为一般而言的动态语言定义是程序运行时,允许改变程序结构或变量类型,这种语言称为动态语言.从这个观点看,Perl,Python,Ruby是动态语言,C++,Java,C#不是

  • Java反射技术原理与用法实例分析

    本文实例讲述了Java反射技术原理与用法.分享给大家供大家参考,具体如下: 本文内容: 产生反射技术的需求 反射技术的使用 一个小示例 首发日期:2018-05-10 产生反射技术的需求: 项目完成以后,发现需要增加功能,并且希望增加功能并不需要停止项目运行. 在希望不关停项目运行的情况下,于是考虑到将功能都放到一个单独的项目之外的模块中,每一个功能实现都从这个模块中获取[实际上这个考虑应该是项目开始前就考虑,这个例子可能不是很好].于是就有了反射的产生.(这种思想有点类似工厂模式,如果学过设计

  • Java反射机制概念、原理与用法总结

    本文实例讲述了Java反射机制概念.原理与用法.分享给大家供大家参考,具体如下: 反射机制是什么 反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法:对于任意一个对象,都能够调用它的任意一个方法和属性:这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制. 反射机制能做什么 反射机制主要提供了以下功能: ① 在运行时判断任意一个对象所属的类: ② 在运行时构造任意一个类的对象: ③ 在运行时判断任意一个类所具有的成员变量和方法: ④ 在运行时调用任意一个

  • java反射遍历实体类属性和类型,并赋值和获取值的简单方法

    实例如下: import java.lang.reflect.Field; import java.lang.reflect.InvocationTargetException; import java.lang.reflect.Method; import java.util.Date; /** * 获取实体类型的属性名和类型 * @param model 为实体类 * @author kou 为传入参数 */ public class GetModelNameAndType { public

  • Java反射机制用法总结

    前言 本篇将从以下几个方面讲述反射的知识: class 的使用 方法的反射 构造函数的反射 成员变量的反射 一.什么是class类 在面向对象的世界里,万物皆对象.类是对象,类是java.lang.Class类的实例对象.另外class类只有java虚拟机才能new出来.任何一个类都是Class 类的实例对象.这实例对象有三种表达方式: public class User{ } public class ClassTest{ User u=new User(); //方式1: Class c1=

  • Java基本类型包装类概述与Integer类、Character类用法分析

    本文实例讲述了Java基本类型包装类概述与Integer类.Character类用法.分享给大家供大家参考,具体如下: 基本类型包装类概述 将基本数据类型封装成对象的好处在于可以在对象中定义更多的功能方法操作该数据. 常用的操作之一:用于基本数据类型与字符串之间的转换. 基本类型和包装类的对应 Byte,Short,Integer,Long,Float,Double,Character,Boolean Integer类 为了让基本类型的数据进行更多的操作,Java就为每种基本类型提供了对应的包装

  • 详解Java中ThreadLocal类型及简单用法

    目录 1 基本概念 2 简单使用 3 应用场景 4 底层原理 4.1 set(Object) 4.2 get() 4.3 remove() 4.4 ThreadLocalMap 5 内存泄漏隐患和防止策略 5.1 为什么会发生内存泄漏? 5.2 怎样防止内存泄漏? 1 基本概念 ThreadLocal类提供了线程局部变量.这些变量与普通变量的不同之处在于,每个访问一个变量(通过其get或set方法)的线程都有自己的.独立初始化的变量副本.ThreadLocal实例通常是希望将状态与线程关联起来的

  • 秒懂Java枚举类型(enum)

    理解枚举类型 枚举类型是Java 5中新增特性的一部分,它是一种特殊的数据类型,之所以特殊是因为它既是一种类(class)类型却又比类类型多了些特殊的约束,但是这些约束的存在也造就了枚举类型的简洁性.安全性以及便捷性.下面先来看看什么是枚举?如何定义枚举? 枚举的定义 上述的常量定义常量的方式称为int枚举模式,这样的定义方式并没有什么错,但它存在许多不足,如在类型安全和使用方便性上并没有多少好处,如果存在定义int值相同的变量,混淆的几率还是很大的,编译器也不会提出任何警告,因此这种方式在枚举

  • java反射超详细讲解

    目录 Java反射超详解✌ 1.反射基础 1.1Class类 1.2类加载 2.反射的使用 2.1Class对象的获取 2.2Constructor类及其用法 2.4Method类及其用法 Java反射超详解✌ 1.反射基础 Java反射机制是在程序的运行过程中,对于任何一个类,都能够知道它的所有属性和方法:对于任意一个对象,都能够知道它的任意属性和方法,这种动态获取信息以及动态调用对象方法的功能称为Java语言的反射机制. Java反射机制主要提供以下这几个功能: 在运行时判断任意一个对象所属

随机推荐