Java 反射类型Type的用法说明

网友投稿 289 2022-10-24


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> 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];

Shttp://tring 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.gVgABvdicrout.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 map;

public Map> 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 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>

fieldClazz: interface java.util.Map

isParameterizedType: true

valueTypeName: java.util.Map>

type0: class java.lang.Integer

type1: java.util.HashMap

type1Name: java.util.HashMap

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 list;

private List> matrix;

private Map 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这个类型。

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

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

以获得List的类型为例子

public void getListType() throws NoSuchFieldException {

Field field = Some.class.getDeclaredField("list");

//如果类似于List这样的类型就是一种GenericType

//注意这是一种Type类型

Type type = field.getGenericType();

if (type instanceof ParameterizedType){

//泛型参数类型

ParameterizedType parameterizedType = (ParameterizedType)type;

Type[] actualTypes = parameterizedType.getActualTypeArguments();

//因为List获得第一个泛型参数,因为只有一个,我们取第一个

//如果我们有多个泛型参数,我们可以根据顺序取不同的泛型参数

assertEquals(actualTypes[0],String.class);

//如果获得List这个原始类型呢?

assertEquals(parameterizedType.getRawType(),List.class);

}else{

throw new IllegalStateException();

}

}

4、复杂的嵌套类型

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

例子如下

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

}

}

总结

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


版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。

上一篇:OSI参考模型 网络分层
下一篇:Volatility工具指令篇
相关文章

 发表评论

暂时没有评论,来抢沙发吧~