软件编程
位置:首页>> 软件编程>> java编程>> Java 反射类型Type的用法说明

Java 反射类型Type的用法说明

作者:importsys  发布时间:2023-01-23 04:15:44 

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

来源:https://blog.csdn.net/qq_31027515/article/details/98355164

0
投稿

猜你喜欢

手机版 软件编程 asp之家 www.aspxhome.com