欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Java 反射類型Type的用法說明

 更新時間:2021年05月19日 14:23:20   作者:importsys  
這篇文章主要介紹了Java 反射類型Type的用法說明,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教

各個方法

1. 得到class的成員變量

首先得到object的class對象

然后在class對象中用getDeclaredFields()方法來獲得class的成員變量

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

2. field的函數(shù)

Field對象有很多成員方法

getName()獲取名稱。

getGenericType() 返回一個Type對象

getType() 返回Class對象

getGenericType 和getType區(qū)別:

返回類型一個是Class對象一個是Type接口。

如果屬性是泛型,getType()返回屬性的接口類型。getGenericType()還能返回參數(shù)類型。

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

結(jié)果:

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

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中如何使用反射獲取類型信息?

最近寫了大量需要根據(jù)類屬性的類型反射注入值的代碼,總結(jié)了以下常用的反射技巧:

一個簡單類的例子

在這個類中,有普通的String類型,有數(shù)組類型,有帶泛型的List類型,有嵌套List類型,以及有多個泛型參數(shù)的簡單類,這個類將作為我們后面的內(nèi)容的基礎(chǔ)。我們這一次博客解析如何使用反射獲取到不同屬性的類型值。

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、數(shù)組類型

數(shù)組類型不像其他的類型可以通過isAssignableFrom()函數(shù)來進(jìn)行判斷,他需要使用isArray() 來判斷該type是否是一個數(shù)組類型,然后使用getComponentType() 獲取他的元素的類型

public void queryArrayType() throws NoSuchFieldException {
    Field field = Some.class.getDeclaredField("numbers");
    Class<?> type = field.getType();
    //一般來說,判斷是否是某種類型是可以使用isAssignableFrom
    // 判斷是否是數(shù)組類型比較特殊,要使用isArray()這個函數(shù)
    if (type.isArray()){
        //獲得數(shù)組的類型,使用getComponentType()這個方法
        Class<?> componentType = type.getComponentType();
        assertEquals(componentType,Integer.class);
    }
    else{
        throw new IllegalStateException();
    }
}

3、帶泛型的類型

帶泛型的類型就是類似于List<String>這樣的類型,我們現(xiàn)在的任務(wù)就是獲取到String這個類型。

ParameterizedType表示參數(shù)化的類型,例如Collection這樣的類型。我們可以通過getGenericType()方法獲得該子類,當(dāng)你的類型帶有參數(shù)的時候就會返回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){
        //泛型參數(shù)類型
        ParameterizedType parameterizedType = (ParameterizedType)type;
        Type[] actualTypes = parameterizedType.getActualTypeArguments();
        //因為List<String>獲得第一個泛型參數(shù),因為只有一個,我們?nèi)〉谝粋€
        //如果我們有多個泛型參數(shù),我們可以根據(jù)順序取不同的泛型參數(shù)
        assertEquals(actualTypes[0],String.class);
        //如果獲得List這個原始類型呢?
        assertEquals(parameterizedType.getRawType(),List.class);
    }else{
        throw new IllegalStateException();
    }
}

4、復(fù)雜的嵌套類型

假如是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、多個泛型參數(shù)

與第三個例子相似,只需要使用actualTypes數(shù)組按順序取即可

例子如下

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

總結(jié)

以上總結(jié)了幾種常用的使用反射獲取屬性類型的例子,稍加改造就可以寫自己的工具類了。希望對大家有幫助^_^

相關(guān)文章

  • SpringBoot在RequestBody中使用枚舉參數(shù)案例詳解

    SpringBoot在RequestBody中使用枚舉參數(shù)案例詳解

    這篇文章主要介紹了SpringBoot在RequestBody中使用枚舉參數(shù)案例詳解,本篇文章通過簡要的案例,講解了該項技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下
    2021-09-09
  • Java?DelayQueue實現(xiàn)任務(wù)延時示例講解

    Java?DelayQueue實現(xiàn)任務(wù)延時示例講解

    DelayQueue是一個無界的BlockingQueue的實現(xiàn)類,用于放置實現(xiàn)了Delayed接口的對象,其中的對象只能在其到期時才能從隊列中取走。本文就來利用DelayQueue實現(xiàn)延時任務(wù),感興趣的可以了解一下
    2022-09-09
  • SpringBoot如何使用TestEntityManager進(jìn)行JPA集成測試

    SpringBoot如何使用TestEntityManager進(jìn)行JPA集成測試

    TestEntityManager是Spring Framework提供的一個測試框架,它可以幫助我們進(jìn)行 JPA 集成測試,在本文中,我們將介紹如何使用 TestEntityManager 進(jìn)行 JPA 集成測試,感興趣的跟著小編一起來學(xué)習(xí)吧
    2023-06-06
  • 一文搞懂Mybatis中Mapper配置文件獲取參數(shù)的五種方式

    一文搞懂Mybatis中Mapper配置文件獲取參數(shù)的五種方式

    這篇文章主要介紹了Mybatis中Mapper配置文件獲取參數(shù)的五種方式,文中通過代碼示例講解的非常詳細(xì),對大家的學(xué)習(xí)或工作有一定的幫助,需要的朋友可以參考下
    2024-03-03
  • Mybatis SqlSessionFactory與SqlSession詳細(xì)講解

    Mybatis SqlSessionFactory與SqlSession詳細(xì)講解

    SqlSessionFactory是MyBatis的核心類之一,其最重要的功能就是提供創(chuàng)建MyBatis的核心接口SqlSession,所以我們需要先創(chuàng)建SqlSessionFactory,為此我們需要提供配置文件和相關(guān)的參數(shù)
    2022-11-11
  • Springboot傳參詳解

    Springboot傳參詳解

    這篇文章主要介紹了Springboot傳參的相關(guān)知識,本文通過實例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友參考下吧
    2023-11-11
  • Java List的sort()方法改寫compare()實現(xiàn)升序,降序,倒序的案例

    Java List的sort()方法改寫compare()實現(xiàn)升序,降序,倒序的案例

    這篇文章主要介紹了Java List的sort()方法改寫compare()實現(xiàn)升序,降序,倒序的案例,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2021-03-03
  • springBoot不同module之間互相依賴的實現(xiàn)

    springBoot不同module之間互相依賴的實現(xiàn)

    本文主要介紹了springBoot不同module之間互相依賴的實現(xiàn),不同模塊之間的依賴通常是通過Maven或Gradle來管理的,下面就來介紹一下如何實現(xiàn),感興趣的可以了解一下
    2024-08-08
  • java注解的類型知識點總結(jié)

    java注解的類型知識點總結(jié)

    在本篇文章里小編給大家整理了一篇關(guān)于java注解的類型知識點總結(jié)內(nèi)容,有興趣的朋友們可以學(xué)習(xí)下。
    2021-03-03
  • MyBatis中動態(tài)sql的實現(xiàn)方法示例

    MyBatis中動態(tài)sql的實現(xiàn)方法示例

    這篇文章主要給大家介紹了關(guān)于MyBatis中動態(tài)sql的實現(xiàn)方法,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2018-11-11

最新評論