返回顶部
首页 > 资讯 > 后端开发 > Python >Java 反射类型Type的用法说明
  • 120
分享到

Java 反射类型Type的用法说明

2024-04-02 19:04:59 120人浏览 独家记忆

Python 官方文档:入门教程 => 点击学习

摘要

各个方法 1. 得到class的成员变量 首先得到object的class对象 然后在class对象中用getDeclaredFields()方法来获得class的成员变量 Fi

各个方法

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();
    }
}

总结

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

--结束END--

本文标题: Java 反射类型Type的用法说明

本文链接: https://lsjlt.com/news/126337.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • Java 反射类型Type的用法说明
    各个方法 1. 得到class的成员变量 首先得到object的class对象 然后在class对象中用getDeclaredFields()方法来获得class的成员变量 Fi...
    99+
    2024-04-02
  • java中的反射及其优点说明
    目录java反射及优点java反射机制(Reflection)1.什么是反射?反射有什么作用?2.反射相关的主要API3.什么是Class类4.调用运行时类的指定结构java反射及优...
    99+
    2024-04-02
  • NumPy-ndarray 的数据类型用法说明
    ndarray 的数据类型 数据类型,即 dtype ,也是一个特殊的对象, 它包含了ndarray需要为某一种类型数据所申明的内存块信息(也成为了元数据,即表示数据的数据) dty...
    99+
    2024-04-02
  • tensorflow中的数据类型dtype用法说明
    Tensorflow中,主要有以下几种数据类型(dtype),在旧版本中,不用加tf也能使用。 有符号整型 tf.int8:8位整数。 tf.int16:16位整数。 tf.int3...
    99+
    2024-04-02
  • 反射机制:getDeclaredField和getField的区别说明
    getDeclaredField和getField区别 在做后台开发时实体的固定字段一般会抽取为一个父类,其他类继承该父类,例如主键字段,会放到一个父类中(IdEntity),其他类...
    99+
    2024-04-02
  • Java中FileWriter类的常用方法说明
    FileWriter字符输出流 构造方法 public FileWriter(String fileName) throws IOException 根据给定的文件名构造一个 ...
    99+
    2024-04-02
  • java 枚举类中的valueOf用法说明
    目录枚举类中的valueOf用法先创建一个BasicEnum的接口创建一个枚举类实现BasicEnum接口枚举类valueOf方法的疑问枚举类中的valueOf用法 前言:今天遇到了...
    99+
    2024-04-02
  • java反射调用类的方法是什么
    Java反射调用类的方法可以通过以下步骤实现:1. 获取需要调用方法的类的Class对象,可以使用`Class.forName()`...
    99+
    2023-08-18
    java
  • 怎么使用java反射获取字段类型
    要使用Java反射获取字段类型,可以按照以下步骤进行:1. 获取对应类的Class对象,可以通过`Class.forName()`方...
    99+
    2023-10-10
    java
  • java技巧:反射判断field类型的操作
    JAVA 反射机制中,Field的getModifiers()方法返回int类型值表示该字段的修饰符。 其中,该修饰符是java.lang.reflect.Modifier的静态属性...
    99+
    2024-04-02
  • 隐式类型的类型反射如何工作?
    问题内容 据我了解,go 是静态类型的,通常不进行隐式类型转换。因此,没有显式类型声明的常量是根据首次使用时的要求而定的。 因此,在下面的代码片段中,我希望 n 是 float64,因...
    99+
    2024-02-06
    隐式类型转换
  • java Stream流常见操作方法(反射,类加载器,类加载,反射)
    目录Stream流常见的中间操作方法Stream流中常见的终结操作方法反射类加载器反射概述Stream流常见的中间操作方法 Streamfilter(Predicate predic...
    99+
    2024-04-02
  • java Long类型转为String类型的两种方式及区别说明
    目录java Long类型转为String类型1、Long.ValueOf("String")返回Long包装类型数据2、Long.parseLong("String")返回long...
    99+
    2024-04-02
  • Java反射中的类有哪些
    本文小编为大家详细介绍“Java反射中的类有哪些”,内容详细,步骤清晰,细节处理妥当,希望这篇“Java反射中的类有哪些”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。一、什么是反射java.lang包提供java...
    99+
    2023-06-29
  • java反射之方法反射的基本操作方法
    本文接上文“java反射之获取类的信息方法(推荐)”,利用反射(invoke)来获取一个类中的方法来执行。1、定义一个类,包含三个名称相同,参数不同的方法class A{ public void print(){ System.ou...
    99+
    2023-05-31
    java 反射 方法
  • java中基本数据类型与Object的关系说明
    目录基本数据类型与Object的关系这里举一个例子Object、基本数据类型的包装类Object类基本数据类型基本数据类型与Object的关系 我知道大家是不是都听说过Object是...
    99+
    2024-04-02
  • 如何使用运行时类型反射来转换类型?
    问题内容 我正在尝试使用泛型构建一个函数,它将接口切片转换为 t 类型的切片。 我想出了以下内容: func convertInterfaceArray[T any](input []...
    99+
    2024-02-06
  • php反射类ReflectionClass用法分析
    ReflectionClass是PHP中的一个内置类,用于获取和操作类的反射信息。通过ReflectionClass,我们可以获取类...
    99+
    2023-08-08
    php
  • R语言 Factor类型的变量使用说明
    factor类型的创建 1. factor( ) > credit_rating <- c("BB", "AAA", "AA", "CCC", "AA", "AAA"...
    99+
    2024-04-02
  • pytorch关于Tensor的数据类型说明
    目录关于Tensor的数据类型说明pytorch Tensor变形函数Tensor的排序与取极值Tensor与NumPy转换关于Tensor的数据类型说明 1. 32位浮点型:tor...
    99+
    2024-04-02
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作